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带 着 任务 去 学 习 , 在 编程 环境 中 学 编程 

39 小 时 专业 学 习 视频 、389 个 实战 范例 包 ( 学 习 测试 诊断 、 
有 趣 实践 任务 、 专 业 资源 库 、 在 线 交流 、 学 习 经 验 分 享 、 

项 目 案例 分 享 、 习 题 与 解答 、 源 程序 等 ) 
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内 容 简介 


《PHP+MySQL 开发 实战 ) 从 初学 者 的 角度 讲述 使 用 PHP 语言 结合 MySQL 数据 库 进 行程 序 开发 应 该 掌握 的 各 项 技 
术 ， 内 容 突出 “基础 ”、“ 全 面 ”、“ 深 入 ”的 特点 ， 同 时 强调 “实战 ”效果 。 书 中 在 介绍 技术 的 同时 提供 实例 ， 同 时 
在 各 章 的 结尾 安排 有 实战 ， 通 过 实战 来 综合 应 用 本 章 所 讲解 的 知识 ， 做 到 理论 联系 实际 ; 每 篇 的 最 后 一 章 有 一 个 综合 实 
例 , 通过 一 个 模块 综合 讲解 本 篇 所 讲解 的 知识 内 容 ; 在 本 书 的 最 后 两 章 中 提供 了 两 个 完整 的 项 目 实例 , 讲述 从 前 期 规划 、 
设计 流程 到 项 目 最 终 实施 的 整个 实现 过 程 。 

全 书 共 分 28 章 ， 主 要 内 容 包括 初 识 PHP 环境 搭建 、PHP 语言 基础 、PHP 流程 控制 语句 、 字 符 串 操作 与 正则 表达 式 、 
初探 数组 、 日 期 和 时 间 的 管理 、 程 序 调试 与 异常 处 理 、 综 合 实例 (一) 在 线 论坛 、MySQL 数据 库 、MySQL 存储 引擎 
与 运算 符 、MySQL 函数 之 选 、MySQL 基本 操作 、MySQL 数据 查询 、 综 合 实例 (二 ) 一 一 留言 本 、MySQL 存储 过 程 和 函 
数 、MySQL 事务 、 触 发 器 、 综 合 实例 (三 ) 物流 管理 系统 、ADODB 类 库 、 数 据 库 编程 技术 、PDO 数据 库 抽象 层 、 
综合 实例 (四 ) BCTY365 网 上 社区 、Smarty 模板 技术 、ThinkPHP 框架 、Zend Framework 框架 、 综 合 实例 (五 ) 一 一 电 
子 商 务 网 站 、 易 查 供求 信息 网 和 图 书馆 管理 系统 。 所 有 知识 都 结合 具体 实例 进行 介绍 ， 对 涉及 的 程序 代码 给 出 了 详细 的 
注释 ， 读 者 可 以 轻松 领会 PHP+MySQL 程序 开发 的 精 蓄 ， 快 速 提高 开发 技能 。 本 书 特色 及 丰富 的 学 习 资 源 包 如 下 : 

黄金 学 习 搭 配 、 专 业 学 习 视频 、 重 难点 精确 打击 、 学 习 经 验 分 享 、 学 习 测试 诊断 、 有 趣 实践 任务 、 专 业 资 源 库 、 学 
习 排忧解难 、 获 取 源 程序 、 提 供 习题 答案 、 赠 送 开发 案例 。 

本 书 适合 有 志 于 从 事 软 件 开发 的 初学 者 、 高 校 计算 机 相关 专业 学 生 和 毕业 生 ， 也 可 作为 软件 开发 人 员 的 参考 手册 ， 
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PHP 和 MySQL 是 全 球 最 普及 、 应 用 最 广泛 的 互联 网 开发 语言 及 最 流行 的 数据 库 之 一 。 PHP 语言 具有 简 
单 、 易 学 、 源 码 开 放 ， 可 操纵 多 种 主流 与 非 主 流 的 数据 库 ， 支 持 面向 对 象 的 编程 ， 支 持 多 种 开源 框架 〈 如 
Zend Framework) ， 支 持 跨 平台 的 操作 ， 而 且 完 全 免费 等 特点 ， 而 MySQL 越 来 越 受 到 广大 程序 员 的 青睐 和 
认同 。 全球 最 大 的 网 络 搜索 引擎 公司 Google 使 用 的 数据 库 就 是 MySQL, 并 且 国 内 很 多 大 型 网 络 公司 也 选择 
MySQL 数据 库 ， 而 PHP 目前 拥有 几 百 万 个 用 户 ,其 发 展 速度 要 快 于 在 它 之 前 的 任何 一 种 计算 机 语言 。 相 信 
PHP 和 MySQL 一 定 能 够 经 得 起 实践 的 检验 ， 发 展 成 为 互联 网 开发 语言 中 “主流 的 主流 ”。 


本 书 特色 及 配套 学 习 资 源 包 


为 了 方便 读者 学 习 ， 本 书 经 过 了 科学 安排 ， 并 配备 了 丰富 的 学 习 资 源 包 ， 读 者 朋友 可 从 本 书 的 配 书 光 
扫 或 者 网 站 www.rjkflm.com 获取 学 习 资源 。 


黄金 学 习 搭配 专业 学 习 视频 重 难点 精确 打击 
快速 入 门 + 中 小 实例 实战 + 模块 光盘 含 39 小 时 大 型 同步 教学 后 305 个 精彩 实例 分 析 ， 精 确 掌 
实战 + 项 目 实战 + 开发 资源 包 。 视频 ， 听 专家 现场 演示 讲解 。 1 担 重 点 难点 。 (图 书 ) 
(图 书 + 光 盘 + 网 站 ) (光盘 中 ) 
学 习 分 享 经 验 学 习 测试 、 诊 断 有 趣 实 践 任务 
提供 互动 、 互 助 学 习 平台 , 学习 @ 网 站 提供 编程 能 力 测试 、 软 件 光盘 提供 580 多 个 实践 任务 ， 
| 分 享 经 验 。 (登录 网 站 ) 7 全 ”考试 模拟 测试 题库 。 读者 可 以 登录 网 站 获取 答案 。 
(登录 网 站 ) (光盘 + 网 站 ) 
专业 资源 库 学 习 排 忧 解难 获取 源 程序 
免费 赠送 PHP 程序 开发 资源 库 | [三 | 提供 编程 学 习 论坛 , 头脑 风暴 ， 光盘 提供 几乎 所 有 的 实例 源 
sl (学 习 版 ) ， 拓 展 编程 视野 。 二 | 帮 您 轻松 解决 编程 困扰 。 ©@. 程序 , 可 直接 复制 , 照 猫 画 虎 ， 
《登录 网 站 ) (登录 网 站 ) 调试 运行 。〈 光 盘 中 ) 
提供 习题 答案 赠送 开发 案例 
过。 本 书 对 于 习题 都 给 出 了 答案 , 先 Ee=， 赠送 开发 案例 文档 、 源 程序 和 
人 自行 作业 ， 然 后 对 比分 析 。 ” 学 习 视频 , 帮助 读者 拓展 视野 ， 
(光盘 中 ) 提高 熟练 度 。〈 光 盘 中 ) 
读者 对 象 
回 ”有志 于 从 事 软件 开发 的 初学 者 回 ”高 等 院 校 计算 机 相关 专业 的 老师 和 学 生 
回 “准备 从 事 软件 开发 的 求职 者 回 ”参与 毕业 设计 的 学 生 
回 “” 初 、 中 级 程序 开发 人 员 加 ”程序 测试 及 维护 人 员 
本 书 内 容 结构 


从 初学 程序 开发 的 人 员 步 入 编程 高 手 行列 通常 需要 经 历 6 个 阶段 ， 即 新 手 入 门 一 初级 开发 一 中 级 开发 
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一 进 阶 提高 一 高 级 应 用 一 项 目 实战 ， 而 本 书 中 的 内 容 正 是 按照 这 一 规律 精心 组 织 的 ， 结 构 如 下 图 所 示 。 


项 目 规划 、 系 统 分 析 、 系 统 设计 到 项 目 实 
施 ， 全 程 进行 讲解 ， 附 有 图 解 和 视频 录像 


第 1 篇 : 新 手 入 门 。 其 主要 内 容 包括 初 识 PHP 环境 搭建 、PHP 语言 基础 、PHP 流程 控制 语句 、 字 符 串 
操作 与 正则 表达 式 、 初 探 数组 、 日 期 和 时 间 的 管理 、 程 序 调试 与 异常 处 理 、 综 合 实例 (一) 在 线 论坛 。 

第 2 篇: 初级 开发 。 其 主要 内 容 包 括 MySQL 数据 库 、MySQL 存储 引擎 与 运算 符 、MySQL 函数 之 选 、 
MySQL 基本 操作 、MySQL 数据 查询 、 综 合 实例 (二 ) 一 一 留言 本 。 

第 3 篇 : 中 级 开发 。 其 主要 内 容 包括 MySQL 存储 过 程 和 函数 、MySQL 事务 、 触 发 器 、 综 合 实例 
(三 ) 一 一 物流 管理 系统 。 

第 4 篇 : 进 阶 提 高 。 其 主要 内 容 包 括 ADODB 类 库 、 数 据 库 编程 技术 、PDO 数据 库 抽象 层 、 综 合 实例 
(四 ) 一 一 BCTY365 网 上 社区 。 

第 5 篇: 高 级 应 用 。 其 主要 内 容 包 括 Smarty 模板 技术 、ThinkPHP 框架 、Zend Framework 框架 、 综 合 
实例 (五 ) 一 一 电子 商务 网 站 。 

第 6 篇 : 项 目 实战 。 通 过 两 个 完整 的 项 目 实例 介绍 了 大 型 应 用 程序 的 设计 过 程 ， 包 括 易 查 供求 信息 网 
和 图 书馆 管理 系统 。 这 两 个 项 目 是 作者 精心 挑选 的 ， 内 容 覆 盖 多 个 方面 。 通 过 对 这 两 个 项 目的 学 习 ， 读 者 
可 以 巩固 前 面 所 学 的 知识 和 技术 ， 积 累 项 目 开发 经 验 。 


本 书 备用 服务 


如 果 本 书 服务 网 站 www.rjkflm.com 临时 有 问题 , 读者 朋友 还 可 以 通过 如 下 方式 与 我 们 沟通 : 登录 网 站 : 
www.mingribook.com， 查 阅 相关 问题 或 者 留言 。 通 过 QQ: 4006751066。 

本 图 书 光盘 如 有 打 不 开 现象 ， 请 核实 一 下 电脑 是 不 是 DVD 光驱 ; 如 果 在 复制 光盘 内 容 时 ， 出 现 个 别 文 
件 无 法 复制 ， 请 分 批复 制 试 一 试 ， 如 有 极 个 别 光盘 打 不 开 ， 可 多 试 几 台电 脑 ， 打 开 之 后 复制 内 容 一 样 使 用 。 


“宝剑 锋 从 磨 研 出 ， 梅 花香 自 苦寒 来 ”， 亲 爱 的 读者 朋友 ， 希 望 在 辛苦 的 道路 上 我 们 一 起 走 过 ! 
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( 种 视频 讲解 : 66 分 钟 ) 


要 使 用 PHP， 首 先 要 建立 PHP 开发 环境 。PHP 是 全 球 最 普及 、 
应 用 最 广泛 的 互联 网 开发 语言 之 一 。 学 习 任 何 一 门 编程 语言 ， 在 开始 
学 习 前 都 要 首先 学 会 搭建 和 熟悉 开发 环境 。 本 章 将 介绍 两 种 操作 系统 
(Windows 和 Linux) 下 的 Apache 服务 器 、My5QL 服务 器 及 PHP 
的 安装 和 配置 方法 。 另 外 ， 还 将 别 为 初学 者 介绍 一 种 简化 安装 和 配置 
PHP 环境 的 组 合 包 。 最 后 ， 使 用 Dreamweaver 开发 PHP 的 第 一 个 
实例 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 


MW 掌握 如 何在 Windows 和 Linux 下 安装 和 配置 Apache 服务 路 
Hi 掌握 如 何在 Windows 和 Linux 下 安装 PHP 

H 掌握 如 何 使 用 组 合 包 快速 搭建 PHP 环境 

WI 掌握 如 何 应 用 开发 工具 编写 、 发 布 和 运行 第 一 个 PHP 程序 
WI 掌握 如 何 独立 解决 常见 的 PHP 环境 安装 问题 
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1.1 PHP 开发 环境 的 搭建 


1.1.1 在 Windows 下 搭建 PHP 开发 环境 


在 Windows 下 搭建 PHP 与 安装 其 他 的 一 些 软件 工具 不 同 。 因 为 PHP 是 从 Linux 移植 过 来 的 一 种 语言 ， 
不 仅 在 开发 环境 上 保留 着 Linux 的 特点 (Apache 是 Linux 下 的 Web 服务 器 ， 地 位 就 像 Windows 下 的 IS; 
MySQL 也 是 Linux 系统 中 捆绑 的 数据 库 ) ， 在 安装 上 也 被 烙 上 了 Linux 印记 。 除 了 正常 的 安装 操作 外 ， 还 
需要 在 各 自 的 配置 文件 〈.ini、.conf) 中 进行 专门 的 设置 。 

安装 之 前 要 准备 的 安装 包 有 : 
Apache 2.2.8-win32-x86-no_ssl.msi。 下 载 地 址 为 http://httpd.Apache.org/download.cgi。 
php-5.2.5-Win32.zip。 下 载 地 址 为 http://www.php.net/downloads.php。 
mysql-essential-5.0.51a-win32.msi。 下 载 地 址 为 http://www.mysql.com/download/( 下载 MySQL 需要 
注册 一 个 账号 ) 。 


1.1.2 在 Linux 下 搭建 PHP 开发 环境 


加 回 加 


在 Linux 下 搭建 PHP 环境 比 Windows 下 要 复杂 得 多 。 除 了 Apache、PHP 等 软件 外 ,还 要 安装 一 些 相关 
工具 ， 设 置 必要 的 参数 。 而 且 ， 要 使 用 PHP 扩展 库 ， 还 要 进行 编译 。 如 本 书 中 使 用 到 的 SOAP、MHASH 
等 扩展 库 。 这 里 给 出 在 Linux 下 安装 的 必要 步骤 。 如 果 用 户 在 安装 过 程 中 遇 到 特殊 的 问题 , 还 需要 翻阅 Linux 
相关 的 书籍 和 手册 。 

安装 之 前 要 准备 的 安装 包 有 : 
httpd-2.2.11.tar.gz。 下 载 地 址 为 http://www.apache.org。 
php-5.2.5.tar.gz。 下 载 地 址 为 http://www.php.net/downloads.php。 
mysql-5.0.51a-Linux-i686.tar.gz。 下 载 地 址 为 http://www.mysql.com。 
libxml2-2.6.26.tar.gz。 可 在 网 络 上 直接 搜索 该 版 本 进行 下 载 。 


图 加 图 加 


1.2 ”Apache 服务 器 的 安装 和 配置 


多 视频 讲解 : 光盘 \TM\Video\ 第 1 章 \Apache 服务 器 的 安装 和 配置 .exe 


1.2.1 在 Windows 下 安装 Apache 服务 器 


Apache 服务 器 是 全 球 范围 内 使 用 范围 最 广 的 Web 服务 软件 , 超过 50% 的 网 站 都 在 使 用 Apache 服务 器 ， 
它 以 高 效 、 稳 定 、 安 全 、 免 费 〈 最 重要 的 一 点 ) 的 优势 成 为 了 最 受 欢迎 的 服务 器 软件 。 
本 节 主 要 介绍 如 何在 Windows 操作 系统 中 安装 和 配置 Apache 服务 器 。 安 装 Apache 服务 器 前 ， 应 到 官 
方 网 站 http://www.apache.org 下 载 Apache 的 安装 程序 Apache 2.2.11-win32-x86-no_ssl.msi。 
在 Windows 下 安装 和 配置 Apache 服务 器 的 操作 步骤 如 下 : 
(1) 双击 Apache 2.2.11-win32-x86-no_sslLmsi 文件 ， 弹 出 欢迎 页 面 。 单 击 Next 按钮 ， 进 入 到 许可 协议 
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页 面 。 
(2) 在 许可 协议 页 面 ， 用 户 需要 同意 页 面 中 的 条 款 才 能 继续 安装 。 选 中 Iaccept the terms in the license 
agreement 单 选 按钮 ， 单 击 Next 按钮 进入 到 下 一 页 面 ， 如 图 1.1 所 示 。 

(3) 图 1.1 所 示 页 面 是 对 该 程序 的 一 个 描述 和 说 明 。 在 了 解 了 相关 的 信息 后 ， 单 击 Next 按钮 进入 到 
Server Information 页 面 。 

(4) Server Information 页 面 需 要 用 户 填写 域名 、 服 务 器 名 称 和 管理 员 Email， 如 图 1.2 所 示 。 该 页 面 还 
有 两 个 单 选 按钮 ， 如 果 选 中 默认 的 第 一 个 单 选 按钮 ， 说 明 该 服务 器 对 所 有 人 开放 ， 并 且 服 务 器 的 端口 号 为 
80， 这 是 推荐 选项 。 选 中 第 二 个 单 选 按钮 是 指 该 服务 器 仅 对 当前 用 户 开 放 ， 并 且 服 务 器 端口 为 8080。 这 里 
选中 第 一 个 单 选 按钮 ， 然 后 单 击 Next 按钮 进入 下 一 个 页 面 。 
ae rr seer22 malon Vin | 国 poemmsmen nasamwa 区 本 | 
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图 1.1 程序 描述 和 说 明 页 面 图 1.2 Server Information 页 面 


《注意 如 果 用 户 的 机 器 安装 有 Internet 信息 服务 (IIS ) 管理 器 ， 那 么 必须 将 此 项 服务 停止 ， 因 为 HIS 
服务 器 的 默认 端口 号 为 80， 同 Apache 服务 器 默认 端口 号 相同 。 如 果 IIS 服务 不 停止 ， 就 会 和 Apache 服 
务 器 的 端口 号 产生 冲突 ，Apache 服务 器 将 不 能 成 功 安装 。 


(5) 如 图 1.3 所 示 的 页 面 用 于 选择 安装 类 型 。 安 装 类 型 分 为 典型 安装 和 自 定义 安装 ， 通 常 保持 默认 设 
置 即 可 。 单 击 Next 按钮 ， 进 入 到 路 径 选 择 页 面 。 

(6) 在 路 径 选 择 页 面 中 ， 单 击 Change 按钮 可 以 选择 安装 路 径 。 这 里 路 径 设 为 DApache2.2\， 如 图 1.4 
所 示 。 
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(7) 单 击 Next 按钮 进入 文件 安装 页 面 。 这 是 Apache 安装 的 最 后 一 步 ， 程 序 开始 安装 文件 。 安 装 结束 
后 ， 单 击 Finish 按钮 结束 安装 程序 。 
(8) 安装 完成 后 ，Apache 服务 器 会 自动 开启 。 在 系统 托盘 区 域 将 出 现 一 个 图 标 ， 当 前 Apache 服务 启 
动 时 ， 图 标 样式 为 名， 服务 器 未 启动 时 ， 图 标 样式 为 条 。 

单 击 Apache 服务 器 的 启动 小 图 标 ， 将 会 看 到 服务 器 的 开启 与 关闭 功能 ， 也 可 以 右 击 小 图 标 ， 在 弹出 的 
快捷 菜单 中 选择 OpenApacheMonitor 命令 ， 打 开 Apache 监控 程序 ， 其 操作 效果 如 图 1.5 所 示 。 

(9) 服务 器 开启 后 ， 需 要 进行 测试 。 打 开 正 浏览 器 页 面 ， 在 地 址 栏 中 输入 “http://127. 0.0.1/” 或 
“http://localhost/”， 按 Enter 键 后 系统 会 显示 如 图 1.6 所 示 的 页 面 ， 此 时 说 明 Apache 服务 器 安装 成 功 。 
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图 1.5 Apache 控制 菜单 图 1.6 ”Apache 服务 器 运行 页 面 


(10) Apache 服务 器 安装 成 功 后 ， 接 下 来 需要 对 Apache 服务 器 进行 配置 ， 以 便 Apache 服务 器 能 够 识 
别 PHP 文件 。 配 置 Apache 服务 器 主要 是 在 Apache 安装 目录 下 的 conf 子 目 录 的 httpd.conf 文件 中 进行 ， 找 
到 该 文件 并 用 记事 本 等 文本 编辑 器 打开 该 文件 。 
(11) 定位 到 LoadModule 配置 块 ， 在 LoadModule 的 最 后 添加 如 下 信息 。 
LoadModule php5_module d:\php5\php5Apache2_2.dll 
添加 后 的 文件 效果 如 图 1.7 所 示 。 


图 1.7 ”加载 .dll 文件 


(12) 修改 DocumentRoot 参数 可 以 修改 Apache 服务 器 主 文档 的 根 目 录 。 原 根 目 录 的 位 置 是 
Apache2.2\htdocs， 用 户 可 以 任意 指定 位 置 。 如 : 
DocumentRoot "D:/Webpage" 
在 DocumentRoot 的 下 面 间隔 约 28 行 的 位 置 ， 有 一 行 “<Directory "D:/Apache2.2/htdocs">”， 修 改 为 
“<Directory "D:/Webpage">”。 


Ap 


和 注意 DocumentRoot 和 这 里 的 参数 值 要 保持 一 致 。 
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(13) 添加 Apache 服务 器 能 够 识别 的 PHP 扩展 名 。PHP 的 扩展 名 有 .php3、.php4、.php、.phtml 等 。 
这 里 只 推荐 使 用 标准 的 扩展 名 .php。 添 加 的 代码 如 下 : 
AddType application/x-httpd-php .php 
添加 位 置 如 图 1.8 所 示 。 
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icnyx-hetpd-php -pbp 


T ERT Enmll i 了 
图 1.8 添加 PHP 扩展 识别 


(14) 默认 显示 页 。Apache 的 默认 显示 页 为 index.html。 也 就 是 说 ， 在 服务 器 未 指名 文件 时 ， 首 先 查找 
index.html， 如 果 找 到 index.html， 那 么 服务 器 就 将 加 载 该 文件 ， 和 否则 显示 目录 内 的 文件 列表 。 在 这 里 添加 一 
个 PHP 默认 页 : index.php。 更 改 后 的 代码 如 下 : 

DirectoryIndex index.html index.php 

(15) 修改 Apache 端口 号 。Apache 的 端口 号 为 80。 修 改 Listen 选项 的 值 ， 即 可 修改 端口 号 。 如 改 为 

82， 则 更 改 后 的 代码 如 下 : 
Listen 82 
以 上 配置 完成 后 ， 重 启 Apache 服务 器 即 可 。 


Ce 
解决 的 办 法 是 改变 其 中 的 一 个 端口 号 ， 或 者 停止 TIS 服务 器 。 


1.2.2 在 Linux 下 安装 Apache 服务 器 


在 Linux 下 安装 Apache 需要 到 官方 网 站 http://www.apache.org 下 载 Linux 下 的 httpd2.2.8.tar.gz 压缩 包 。 
首先 需要 打开 Linux 终端 (Linux 系统 中 ， 几 乎 所 有 的 软件 都 需要 在 终端 下 安装 ) 。 在 RedHat9 的 界面 
中 选择 “ 主 菜单 ”/“ 系 统 工具 ”命令 ， 在 弹出 的 菜单 中 选择 “终端 ”命令 。 
在 Linux 下 安装 和 配置 Apache 服务 器 的 操作 步骤 如 下 : 
(1) 进入 到 Apache 安装 文件 的 目录 下 ， 如 /usrlocal/work。 
cd /usr/local/work/ 
(2) 解压 安装 包 。 完 成 后 进入 到 httpd2.2.8 目录 中 。 
tar xfz httpd2.2.8.tar.gz 
cd httd2.2.8 
(3) 建立 makefile， 将 Apache 服务 器 安装 到 user/local/Apache2 下 。 
.configure —prefix=/usr/local/Apache2 -enable-module=so 
(4) 编译 文件 。 
make 
(5) 开始 安装 。 
make install 


(6) 安装 完成 后 ， 将 Apache 服务 添加 到 系统 启动 项 中 ， 重 启 服务 器 。 
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/usrlocal/Apache2/bin/Apachectl start >> /etc/rc.d/rc.local 
/usrlocal/Apache2/bin/Apachectl restart 


(7) 打开 Mozilla 浏览 器 ， 在 地 址 栏 中 输入 “http:Wlocalhosy”， 按 Enter 键 后 如 果 看 到 如 图 1.9 所 示 的 
页 面 ， 说 明 Apache 服务 器 已 安装 成 功 。 
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训 训 et De 可 范 - 贺 
失主 页 奈 放 入 可 Rod Ha Nat 拓 Sqpm 前 shr 痢 hoiee 莉 Tnias 

知 能 看 见 这 个 页 面 ， 说 明 &pache meb server 已 经 安装 记功 。 您 可 以 在 这 个 目 好 中 增加 内 

容 ， 或 者 把 这 个 页 面 谷 换 掉 - 

这 不 是 你 想 看 见 的 页 面 吧 ? 

之 所 以 会 竺 见 这 个 页 面 ， 是 因为 网 站 管理 由 改 充 了 这 个 站 点 的 设置 ， 如 采 有 闯 辣 ， 靖 咨询 维 

护 此 由 点 的 人 员 。 Apachc 软 件 基 全 会 ， 即 此 竺 点 所 侥 用 的 网 站 服务 瑟 软 件 的 攻 发 者 ， 不 负 

位 此 引 点 的 维护 工作 ， 也 无 法 为 如 解 决 设置 上 的 问题 

pache 文档 已 经 包含 在 此 发 行 乒 中 - 

您 可 以 在 使 用 Apache 的 网 站 服务 话 上 ， 自 由 地 使 用 下 面 的 图 片 。 感 谢 使 用 Apachel 
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图 1.9 Linux 下 的 Apache 服务 器 安装 页 面 
1.3 PHP 的 安装 和 配置 


卫视 频 讲 解 : 光盘 \TM\Video\ 第 1 章 \PHP 的 安装 和 配置 .exe 
1.3.1 在 Windows 下 安装 PHP 


架设 基于 PHP 的 Web 服务 器 ， 必 须 安装 PHP。 由 于 
PHP 的 代码 公开 , 所 以 其 升级 速度 较 快 。 安 装 PHP 之 前 应 
从 其 官方 网 站 http://www.php.net/ 下 载 最 新 版 本 的 PHP 安 
装 程序 php-5.2.5-Win32.zip。 

Apache 服务 器 顺利 启动 后 ， 接 下 来 安装 PHP5。 在 
Windows 下 安装 和 配置 PHP 的 操作 步骤 如 下 : 

(1) 将 PHP5 的 安装 文件 php-5.2.5-Win32.zip 解压 到 
相应 目录 ， 如 C:\php、E:\php5 等 。 这 里 将 其 放 到 E:\php5 
目录 下 。 目 录 结 构 如 图 1.10 所 示 。 

(2) 将 该 目录 下 的 所 有 .dll 文件 复制 到 系统 盘 
Windows\system32 目录 下 (Windows 2000 是 在 winnt\ A 
system32 目录 下 ) 。 

(3) 将 php.ini-dist 文件 复制 到 系统 盘 \Windows 目录 图 1.10 PHP5 的 目录 结构 
下 ， 并 重新 命名 为 php.ini。 

(4) 打开 php.ini 文件 并 找到 “extension di ="/"”， 修 改 为 “extension dir ="e:/php5/ext"”。 

(5) 找到 “;extension=php_mysql.dll”， 将 前 面 的 分 号 “;” 去 掉 。 这 样 ，PHP 即 可 支持 MySQL 数 
据 库 。 

(6) PHP 配置 完成 以 后 ， 重 新 启动 Apache 服务 器 。 

(7) 编写 一 个 PHP 脚本 文件 , 命名 为 phpinfo php, 保存 在 Apache 服务 器 的 虚拟 目录 D:\htdocs 下 。PHP 
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脚本 文件 的 代码 如 下 : 

<?php 

phpinfo(); // 获 取 PHP 的 配置 信息 

站 

(8) 最 后 在 浏览 器 的 地 址 栏 中 输入 “http://localhost/phpinfo.php”， 如 果 显示 PHP 的 版 本 相关 信息 ， 则 
说 明 PHP 配置 成 功 。 


1.3.2 在 Linux 下 安装 PHP 


安装 PHP5 之 前 ， 需 要 首先 查看 libxml 的 版 本 号 。 如 果 libxml 版 本 号 小 于 2.5.10， 则 需要 先 安装 libxml 
高 版 本 。 安 装 libxml 和 PHP5 的 步骤 如 下 〈 如 果 不 需 要 安装 libxml， 则 直接 跳 到 PHP5 的 安装 步骤 即 可 ) : 
(1) 将 libxml 和 PHP5 复制 到 /usr/local/work/ 目 录 下 ， 并 进入 到 该 目录 。 
cp php-5.2.5.tar.gz libxml2-2.6.26.tar.gz /usr/local/work 
cd /usr/local/work 
(2) 分 别 将 libxml2 和 PHP 解压 。 
tar xfz libxml2-2.6.62.tar.gz 
tar xfz PHP-5.2.5.tar.gz 
(3) 进入 到 libxml2 目录 ， 建 立 makfile， 将 libxml 安装 到 /usrlocallibxml2 下 。 
cd libxml2-2.6.62 
.configure —prefix=/usr/local/libxml2 
(4) 编译 文件 。 
makefile 
(5) 开始 安装 。 
make install 
(6) libxml2 安装 完毕 ， 开 始 安装 PHP5。 进 入 到 php-5.2.5 目录 下 。 
cd ./php-5.2.5 
(7) 建立 makefile。 
./configure —with-apxs2=/usr/local/Apache2/bin/apxs 
--With-mysql=/usr/local/mysql 
with-libxml-dir=/usr/local/libxml2 
(8) 开始 编译 。 
make 
(9) 开始 安装 。 
make install 
(10) 复制 php.ini-dist 或 php.ini-recommended 到 /usrlocallib 目录 ， 并 命名 为 php.ini。 
cp php.ini-dist /usr/local/lib/php.ini 
(11) 更 改 httpd.conf 文件 相关 设置 ,该 文件 位 于 /usr/local/Apache2/conf 中 。 找到 该 文件 中 的 如 下 指令 行 : 
AddType application/x-gzip .gz .tgz 
在 该 指令 后 加 入 如 下 指令 : 
AddType application/x-httpd-php .php 
重新 启动 Apache， 并 在 Apache 主 目录 下 建立 文件 phpinfo.php。 
<?php 
phpinfo(); 
> 
在 Mozilla 浏览 器 的 地 址 栏 中 输入 “http://localhost/phpinfo.php”， 如 果 出 现 如 图 1.11 所 示 的 页 面 ， 说 
明 PHP 安装 成 功 。 
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图 1.11 phpinfo 信息 
1.4 应 用 组 合 包 快 速 搭建 PHP 环境 


狂 0 视频 讲解 : 光盘 \TM\Video\ 第 1 章 \ 应 用 组 合 包 快 速 搭建 PHP 环境 .exe 

组 合 包 ， 就 是 将 Apache、PHP 和 MySQL 等 服务 器 软件 和 工具 安装 配置 完成 后 打包 处 理 。 开 发 人 员 只 
要 将 已 配置 的 套件 解压 到 本 地 硬盘 中 即 可 使 用 , 无 须 再 另行 配置 。 组 合 包 实现 了 PHP 开发 环境 的 快速 搭建 。 
对 于 初学 PHP 的 程序 员 ， 建 议 采 用 此 方法 搭建 PHP 的 运行 环境 。 组 合 包 安 装 简单 、 速 度 较 快 、 运 行 稳定 ， 
能 使 用 户 将 精力 更 好 地 集中 到 PHP 的 学 习 中 。 

目前 网 上 流行 的 组 合 包 有 十 几 种 ， 安 装 方法 基本 相同 。 这 里 推荐 3 种 组 合 包 : EasyPHP、AppServ 和 
XAMPP。 

EasyPHP 的 下 载 地 址 为 http://www.easyphp.org/。 

AppServ 的 下 载 地 址 为 http://www.appservnetwork.com/。 

回 XAMPP 的 下 载 地 址 为 http://www.Apachefriends.org/。 

建议 新 手 使 用 EasyPHP 或 AppServ, 两 者 都 是 Apache+MySQL+PHP 开发 环境 的 。 而 XAMPP 则 相对 复 
杂 一 些 ， 不 仅 可 以 切换 PHP4 和 PHP5， 还 集成 了 perl 开发 环境 、 第 三 方 扩展 库 等 ， 并 且 对 开发 平台 进行 了 
优化 和 整理 。 如 果 对 PHP 的 开发 环境 已 经 有 了 一 定 的 了 解 ， 则 推荐 使 用 XAMPP。 


ne 
载 ， 再 开始 安装 组 合 包 。 组 合 包 的 安装 很 简单 ， 只 要 将 程序 解压 或 安装 到 指定 目录 就 可 以 直接 使 用 。 


本 节 以 AppServ 组 合 包 的 AppServ-win32-2.5.10 版 本 为 例 ， 重 点 讲解 该 组 合 包 的 安装 和 使 用 方法 。 
AppServ 是 PHP 网 页 架 站 工具 组 合 包 ， 是 将 网 络 上 免费 的 架 站 资源 重新 包装 成 单一 的 安装 程序 。 它 提 
供 了 简易 、 快 速 的 PHP 运行 环境 的 搭建 机 制 ， 读 者 只 需 按照 普通 应 用 软件 的 安装 方式 就 可 以 完成 
Apache+MySQL+PHP+phpMyAdmin 的 安装 与 配置 工作 。 可 以 说 ，AppServ 是 初学 者 的 首选 。 
安装 AppServ 之 前 应 从 官方 网 站 http://www.appservnetwork.com 下 载 AppServ-win32-2.5.10.exe 安装 程序 。 
在 Windows 下 应 用 AppServ 组 合 包 快速 搭建 PHP 开发 环境 的 操作 步骤 如 下 : 
(1) 双击 AppServ-win32-2.5.10.exe 文件 ， 打 开 如 图 1.12 所 示 的 AppServ 启动 页 面 。 
(2) 单 击 Next 按钮 ， 打 开 如 图 1.13 所 示 的 AppServ 安装 协议 页 面 。 
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(3) 单 击 


装 路 径 一 般 为 


目录 下 。 
(4) 单 
全 选 状态 ) 。 


Welcome to the AppServ 2.5.10 
Setup Wizard 
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图 1.12 ”AppServ 启动 页 面 
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1.14 选择 AppServ 安装 路 径 


击 Next 按钮 将 打开 如 图 1.15 所 示 的 页 面 。 在 该 
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图 1.13 AppServ 安装 协议 页 面 


击 IAgree 按钮 将 打开 如 图 1.14 所 示 的 页 面 。 在 该 页 面 中 可 以 设置 AppServ 的 安装 路 径 〈 默 认 安 
C:\AppServ) ，AppServ 安装 完成 后 ，Apache、MySQL 和 PHP 都 将 以 子 目 录 的 形式 存储 到 该 


页 面 中 可 以 选择 要 安装 的 程序 和 组 件 〈 默 认为 
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1.15 ”AppServ 安装 选项 


击 Next 按钮 ， 打 开 如 图 1.16 所 示 的 页 面 。 该 页 面 主要 用 于 设置 Apache 的 端口 号 。 
f Next 按钮 ， 打 开 如 图 1.17 所 示 的 页 面 。 该 页 面 主要 用 于 对 MySQL 数据 库 的 root 用 户 的 登 


录 密 码 及 字符 集 进行 设置 。 这 里 将 字符 集 设置 为 GB2312 Simplified Chinese， 表 示 MySQL 数据 库 的 字符 集 
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图 1.16 设置 Apache 端口 号 
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图 1.17 设置 MySQL 登录 密码 及 字符 集 
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(7) 单 击 Install 按钮 后 开始 安装 ， 如 图 1.18 所 示 。 


(8) 安装 完成 页 面 如 图 1.19 所 示 。 安 装 完成 后 ， 可 以 在 开始 菜单 的 AppServ 相关 操作 列表 中 启动 
Apache 及 MySQL 服务 。 
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图 1.18 AppServ 安装 进度 页 面 图 1.19 AppServ 安装 完成 页 面 


(9) 安装 好 AppServ 后 ， 整 个 目录 默认 为 C:\AppServ,， 此 目录 下 包含 4 个 子 目 录 ， 如 图 1.20 所 示 ， 用 
户 可 以 将 所 有 网 页 文件 存放 到 www 目录 下 。 


(10) 打开 浏览 器 , 在 正 浏 览 器 的 地 址 栏 中 输入 “http:Wlocalhost” 或 者 “http:/127.0.0.1/”， 如 果 打 开 
如 图 1.21 所 示 的 页 面 ， 则 说 明 AppServ 安装 成 功 。 
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六 Easy way to build Webserver, Database Server with AppServ :-) 


1.21 AppServ 测试 页 


oh, 
mg Apache 的 端口 号 是 82， 那 么 在 下 浏览 器 的 地 址 栏 中 输入 
“http://localhost:82/” 或 者 “http://127.0.0.1:82/” 来 测试 AppServ 是 否 安 装 成 功 。 
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1.5 第 一 个 PHP 程序 


句 4 视频 讲解 :光盘 \TM\Video\ 第 1 章 \ 第 一 个 PHP 程序 .exe 
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1.5.1 使 用 Adobe Dreamweaver 编写 源 程序 


服务 器 环境 配置 完成 后 ， 接 下 来 应 用 Adobe Dreamweaver 开发 工具 来 编写 第 一 个 PHP 程序 。 

例 1.1 编写 第 一 个 PHP 程序 的 目的 是 熟悉 PHP 的 书写 规则 和 Adobe Dreamweaver 工具 的 基本 使 用 方 
法 。 本 实例 很 简单 ， 就 像 初学 大 多 数 语言 一 样 ， 输 出 一 段 欢迎 信息 。 (实例 位 置 : 光盘 \TM\Instance\01\1.1) 

开发 步骤 如 下 : 

(1) 启动 Adobe Dreamweaver 编辑 器 ， 选 择 “ 文 件 ”/“ 新 建 ”命令 ， 打 开 “ 新 建文 要 ”对 话 框 ， 在 
“页 面 类 型 ”列表 框 中 选择 PHP 选项 ， 如 图 1.22 所 示 。 

(2) 单 击 “ 创 建 ” 按 钮 ， 即 可 成 功 创建 一 个 动态 的 PHP 页 面 。 对 新 创建 的 页 面 ， 可 以 在 “代码 ”视图 
中 编辑 PHP 代码 ， 也 可 以 使 用 “设计 ”视图 查看 HTML 效果 。 这 里 使 用 “代码 ”视图 ， 并 给 该 页 面 标题 命 
名 ， 如 图 1.23 所 示 。 标 题 显 示 在 浏览 器 的 左上 角 ， 稍 后 运行 时 就 能 看 到 效果 。 
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[EIT 和 
1.22 “新 建文 档 ” 对 话 框 图 1.23 命名 标题 
(3) 编写 PHP 代码。 在 <body>…</body> 标 签 对 中 编写 PHP 代码 段 。 代 码 如 下 : 
<?php 


echo "欢迎 学 习 PHP 语言 ! ! "; 


?> 

代码 讲解 如 下 : 

加 “<?php” 和 “?>” 是 PHP 的 标记 符 。 在 这 对 标记 符 中 的 所 有 代码 都 被 当 作 PHP 代码 来 处 理 。 除 
了 这 种 表示 方法 外 ，PHP 还 可 以 使 用 ASP 风格 的 “<%” 和 SGML 风格 的 “<?…?>” 等 ， 在 第 3 
章 中 将 会 详细 介绍 。 

echo 语句 是 PHP 中 最 常用 的 语句 ， 主 要 用 于 将 一 个 或 多 个 字符 串 输出 至 网 页 ， 和 ASP 中 的 
response.write 以 及 JSP 中 的 out.print 意思 相同 ， 就 是 将 紧 跟 在 后 面 的 字符 串 或 者 变量 值 显示 在 页 
面 中 。 每 行 代码 都 以 分 号 “;” 结 尾 。 
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ot 在 使 用 echo 语句 时 , 应 该 时 刻 牢记 语句 结束 处 的 分 
号 。 在 PHP 中 ,分 号 是 用 来 分 隔 语 名 的。 丢失 “;” 是 经 常会 
犯 的 语法 错误 ， 同 样 也 是 在 进行 程序 的 错误 处 理 时 ， 首 先 应 
该 注意 的 地 方 。 

在 Adobe Dreamweaver 中 输入 的 PHP 脚本 程序 如 图 1.24 所 
示 。 1.24 在 开发 工具 中 输入 PHP 脚本 程序 
(4) 将 PHP 动态 页 保存 到 服务 器 指定 的 目录 以 便 解析 。 本 
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章 中 服务 器 指定 的 目录 为 E:\appservwwww\， 则 将 本 页 保存 到 E:\appserviwww\tm\01\1.\ 路 径 下 ， 命 名 为 
info.php。 


1.5.2 ”发 布 和 运行 PHP 程序 


打开 正 浏 览 器 ,在 地 址 栏 中 输入 “http://127.0.0.1:82/TM/01/1.1/ 
info.php”, 按 Enter 键 后 即 可 查看 info.php 页 的 执行 结果 , 如 图 1.25 
所 示 。 1.25 PHP 页 面 运行 结果 


FETT 全 ”sl0% -| 


1.6 环境 安装 常见 问题 
车 4 视频 讲解 : 光盘 \TM\Video\ 第 1 章 \ 环 境 安装 常见 问题 .exe 


1.6.1 Apache 安装 常见 问题 


1. 解决 Apache 服务 器 端口 冲突 

IIS 服务 器 和 迅雷 的 默认 端口 号 为 80， 同 Apache 服务 器 默认 端口 号 相同 。 由 于 两 者 采用 了 相同 的 端口 
号 80， 因 此 ， 在 运行 网 页 时 就 会 发 生 冲 突 。 

如 果 用 户 安装 了 IS 服务 器 ， 就 需要 修改 IS 的 默认 端口 ， 否 则 将 导致 Apache 服务 器 无 法 正常 工作 。 
更 改 IS 的 默认 侦 听 端口 80， 可 以 在 IS 的 管理 器 中 进行 设置 ， 也 可 以 停止 HS 的 服务 。 

如 果 用 户 安装 并 开启 了 迅雷 软件 ,就 需要 关闭 该 软件 ， 否 则 端口 冲突 将 会 导致 运行 PHP 网 页 程序 时 出 错 。 

用 户 也 可 以 在 安装 Apache 服务 器 时 将 默认 的 端口 号 进行 更 改 ， 从 而 解决 两 个 服务 器 或 与 其 他 软件 共用 

-个 端口 号 而 产生 冲突 的 问题 。 

2. 更 改 Apache 服务 器 默认 存储 的 文件 路 径 

Apache 服务 器 的 核心 配置 文件 是 httpd.conf， 存 放 路 径 为 “Apache 的 安装 路 径 \confN”， 用 记事 本 程序 
打开 该 文件 ， 定 位 到 DocumentRoot。 语 句 如 下 : 

DocumentRoot " D:/Webpage" 

该 语句 用 于 指定 网 站 路 径 ， 也 就 是 主页 放置 的 目录 。 可 以 使 用 默认 路 径 ， 也 可 以 任意 指定 。 需 要 注意 
的 是 ， 语 句 的 末尾 不 要 加 “/”。 

同时 还 要 定位 到 “<Directory ">” 一 行 , 在 双 引 号 中 添加 服务 器 的 虚拟 路 径 , 这 里 要 与 “DocumentRoot” 
一 行 中 设置 相同 。 

<Directory " D:/Webpage "> 


和 注意 路 径 的 分 隔 符 在 Apache 服务 器 里 写成 “/”。 
1.6.2 PHP 安装 常见 问题 


1. PHP 的 安装 路 径 
安装 文件 的 路 径 也 要 遵循 一 定 的 客观 原则 ， 为 了 避免 在 Windows 和 Linux 间 移 植 程序 时 带 来 的 不 便 ， 
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选择 D:vusrlocalyphp 的 目录 时 要 和 在 Linux 下 的 安装 目录 相 匹 配 。 建议 最 好 不 要 选择 中 间 有 空格 的 目录 ,如 
E:\program Files\PHP， 这 样 做 会 导致 发 生 一 些 未 知 错误 甚至 崩溃 。 

2. 控制 上 传 文件 的 大 小 

在 网 站 开发 的 过 程 中 ， 为 了 确保 能 够 充分 利用 服务 器 的 空间 ， 禁 止 上 传 一 些 垃圾 文件 ， 给 网 站 的 维护 
带 来 不 必要 的 麻烦 ， 最 好 对 上 传 文件 的 大 小 进行 限制 ， 将 它 控制 在 有 效 上 传 文件 大 小 的 范围 内 。 如 果 要 在 
PHP 中 实现 小 文件 的 上 传 (2MB 以 下 ) ， 那 么 无 须 对 php.ini 配置 文件 进行 修改 ， 使 用 默认 参数 即 可 。 但 如 
果 想 实现 完美 的 上 传 功能 ， 则 一 定 要 对 php.ini 进行 一 些 修改 。 

Resource Limits， 直 译 就 是 资源 限制 ， 包 含 3 个 参数 。 该 区 块 不 仅 针对 上 传 和 下 载 ， 还 可 对 全 部 的 文件 
进行 设置 。 各 个 参数 含义 及 参数 值 说 明 如 表 1.1 所 示 。 


表 1.1 Resource Limits 块 的 参数 说 明 


每 个 脚本 页 面 完成 执行 操作 的 最 大 时 间 ， 单 位 是 秒 。 如 果 设 为 -1， 说 明 没有 限制 


每 个 脚本 页 面 处 理 请 求 数据 的 最 大 时 间 ， 单 位 是 秒 ， 也 可 以 设 为 -1 
一 个 脚本 页 所 能 够 消耗 的 最 大 内 存 


File Uploads 块 是 专 为 文件 上 传 设置 的 ， 包 含 3 个 参数 ， 参 数 含义 及 参数 值 说 明 如 表 1.2 所 示 。 
表 1.2 File Uploads 块 的 参数 说 明 


是 否 人 允许 HTTP 上 传 ， 默 认为 On， 即 为 开启 ， 无 须 修改 


文件 上 传 时 的 临时 存储 目录 。 如 果 没 指定 就 会 用 系统 默认 的 临时 文件 夹 
load_max_filesize 允许 上 传 的 文件 的 最 大 值 


如 果 想 要 上 传 更 大 的 文件 ， 就 必须 对 上 述 3 个 参数 值 进行 更 改 ， 更 改 后 重新 启动 Apache 服务 器 即 可 。 


(ot er 


1.6.3 ”MySQL 安装 常见 问题 


在 网 站 运作 的 过 程 中 ， 各 类 错误 均 不 可 避免 ， 当 数据 库 连 接 失败 时 ， 除 开启 MySQL 服务 检测 是 否 正常 
运行 外 ， 还 可 以 检查 php.ini 文件 是 否 配置 正确 ， 以 支持 MySQL 服务 。 
打开 C:\Windows\ 目 录 下 的 php.ini 文件 ， 定 位 到 如 图 1.26 所 示 的 代码 位 置 。 


图 1.26 修改 php.ini 文 件 以 支持 MySQL 数据 库 
将 代码 前 面 的 分 号 删除 , 然后 保存 php.ini 文件 , 最 后 重新 启动 Apache 服务 器 , 即 可 让 PHP 支持 MySQL 
在 浏览 器 的 地 址 栏 中 输入 “http://127.0.0.1/phpinfo.php” 或 “http://localhost/phpinfo.php”， 如 果 检 索 到 
MySQL 服务 ， 如 图 1.27 所 示 ， 则 说 明 MySQL 服务 正常 运行 。 
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图 1.27 测试 MySQL 服务 是 否 正常 运行 


1.7 实 战 


例 1.2 应 用 Adobe Dreamweaver 开发 工具 ， 通 过 调用 PHP 的 date0 函 数 ， 动态 显示 系统 时 间 。〔 实 例 
位 置 : 光盘 \ITM\Instance\01\1.2) 


1. 创建 站 点 


编程 的 第 一 步 就 是 创建 Web 站 点 , 在 www 服务 器 的 根 目录 下 创建 指定 的 TM/01/1.2 文件 夹 , 用 来 存储 
PHP 文件 和 相关 资源 。 


2. 使 用 Adobe Dreamweaver 创建 PHP 文件 


启动 Adobe Dreamweaver 编辑 器 ， 选 择 “ 文 件 ”/“ 新 建 ”命令 ， 打 开 “ 新 建文 档 ” 对 话 框 ， 选 择 “ 空 白 
页 ”选项 卡 中 的 “页 面 类 型 ”/“PHP 选项 ”， 然 后 单 击 “ 创 建 ”按钮 ， 即 可 成 功 创建 一 个 动态 的 PHP 页 面 。 
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3. 创建 PHP 标记 

在 <body> 标 记 中 使 用 “<?php ?>” 标 记 符 将 PHP 的 脚本 语言 括 起 来 ， 而 这 些 PHP 脚本 在 浏览 器 中 并 
不 可 见 ， 当 用 户 在 正 浏览 器 中 输入 URL 地 址 “http://localhost/PHP Web/index.php”， 服 务 器 端 接 到 请 求 后 ， 
会 使 用 PHP 的 默认 语言 解释 成 标准 的 HIML 格式 的 代码 传送 到 用 户 的 浏览 器 中 。 


由 5 解释 生成 的 HTML 代码 ， 用 户 可 以 在 下 浏览 嚣 中 选择 “查看 ”/“ 源 文件 ”命令 ， 打 开 程 序 
的 源 代码 ， 它 是 标准 的 HTML 文件 。 
因为 PHP 代码 在 服务 器 端 运行 ， 在 执行 后 才 产 生 浏览 器 能 识别 的 HTML 语言 ， 并 将 其 传递 给 浏览 器 。 
4. 编写 PHP 代码 


在 <body> 标 记 中 输入 如 下 代码 : 
<?php 
echo "<font color='#fTDFDF'> 现 在 时 刻 北 京 时 间 :". date("Y-m-d Hii:s ")."</font>" 
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?> 


代码 解释 如 下 : 

在 <body>…</body> 标 签 对 中 添加 “<?php” 和 “?>” 标 记 符 。 第 一 行 的 “<?php” 是 PHP 代码 段 的 开始 
标记 。 第 三 行 的 “?>” 是 PHP 代码 段 的 结束 标记 。 

第 二 行 的 echo 语句 用 来 输出 字符 串 。 其 中 ，“<font color=#fIDFDF'>…</font>” 代 表 将 包含 在 该 标记 
间 的 字符 串 设置 指定 的 颜色 (这 里 为 粉色 ) 。 调 用 date0 函 数 来 获取 系统 的 当前 时 间 ， 其 中 “Y-m-d H:i:s” 
是 以 指定 的 格式 输出 系统 的 当前 日 期 和 时 间 ， 并 用 字符 串 操作 符 “.” 将 多 个 字符 串 连 接 在 一 起 进行 输出 。 


2 
当前 的 时 间 必 须 更 改 PHP 语言 中 的 时 区 设置 。 更 改 PHP 语言 中 的 时 区 设置 有 两 种 方法 :( 1 ) 修改 php.ini 
文件 中 的 设置 , 找到 [date] 下 的 “:date.timezone =” 选 项 , 将 该 项 修改 为 “date.timezone =Asia/Hong Kong”， 


然后 重新 启动 Apache 服务 器 。 
(2 ) 在 程序 中 , 使 用 时 间 / 日 期 函数 前 添加 函数 “date_default timezone_set("Asia/Hong Kong");” 即 可 。 


在 Adobe Dreamweaver 中 输入 PHP 脚本 程序 ， 如 图 1.28 所 示 。 


5. 存储 文件 
在 创建 好 PHP 文件 后 , 选择 “文件 /保存 ”命令 , 在 弹出 的 “另存 为 "对 话 框 中 选择 已 创建 的 TM/01/1.2 
文件 夹 作 为 PHP 文件 的 保存 位 置 ， 将 文件 命名 为 index.php， 然 后 单 击 “ 保 存 ” 按 钮 ， 如 图 1.29 所 示 。 
EE le 
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图 1.29 “另存 为 ”对 话 框 


图 1.28 在 开发 工具 中 输入 PHP 脚本 程序 


6. 网 站 运行 结果 
打开 正 浏览 器 ， 在 地 址 栏 中 输入 URL 地 址 “http://127.0.0.1:82/tm/01/1.2/index.php”， 按 Enter 键 打开 


该 页 面 ， 运 行 结 果 如 图 1.30 所 示 。 
Te 
En 会 ~ 日 -局 学 ~- P29 I 柳 -“ 


1.30 打开 网 站 页 面 
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本 章 主要 介绍 了 PHP 环境 的 安装 与 配置 方法 ， 读 者 应 重点 掌握 。 通 过 本 章 的 学 习 ， 读 者 可 以 轻松 解决 
环境 配置 的 常见 问题 , 并 可 以 独立 开发 、 发 布 和 运行 简单 的 PHP 程序 。 对 于 本 章 讲 解 的 在 Windows 和 Linux 
系统 上 搭建 PHP 环境 的 方法 ， 读 者 可 以 根据 自身 的 需求 有 选择 性 地 阅读 。 


1.9 学 习 成 果 检 验 


1. 尝试 开发 一 个 页 面 ， 应 用 echo 语句 输出 字符 串 “一 定 要 努力 学 习 PHP 编程 语言 。”。 (答案 位 置 ; 
光盘 \TM\Instance\01\1.3) 


2. 尝试 开发 一 个 页 面 ,应 用 echo 语句 输出 一 条 最 新 公告 信息 ， 要 求 公告 后 面 添加 发 布 日 期 和 时 间 ， 输 
出 效果 如 图 1.31 所 示 。 答案 位 置 : 光盘 \TM\Instance\01\1.4) 


1.31 发 布 最 新 公告 信息 


3. 尝试 开发 一 个 页 面 ， 使 用 echo 语句 输出 一 个 4x3 像素 大 小 的 表格 。 (答案 位 置 : 光盘 \TM\Instance\ 
01\1.5) 
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通过 第 1 章 的 学 习 ， 相 信 读 者 对 PHP 的 概念 和 如 何 搭建 PHP 开 
发 环境 已 有 了 一 个 全 面 的 了 解 ， 本 章 将 学 习 PHP 的 语言 基础 。 无 论 是 
初出 莽 房 的 “ 莱 岛 "还 是 资历 深厚 的 “高 手 ”， 没 有 扎实 的 基础 做 后 
盾 是 不 行 的 。PHP 的 特点 是 易学 、 易 用 ， 但 这 并 不 代表 随 随便 便 就 可 
以 熟练 事 担 。 随 着 知识 的 深入 ，PHP 会 越 来 越 难 学 ， 基 础 的 重要 性 也 
就 越 加 明显 。 训 可 了 基础 ， 就 等 于 有 了 坚 国 的 地 基 ， 才 有 可 能 “万 丈 
高 楼 平地 起 ”。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

Nm 了 解 PHP 的 标记 风格 

MW 了解 PHP 的 数据 类 型 

MW 了 解 PHP 的 常量 应 用 

WI 了 解 PHP 的 变量 应 用 

MH 了 解 PHP 运算 符 

WI 了 解 PHP 的 自 定义 国 数 

Wm 掌握 PHP 的 输出 语句 
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2.1 PHP 语法 基础 


2.1.1 PHP 标记 风格 


名 4 视频 讲解 :光盘 \TM\Video\ 第 2 章 \PHP 标记 风格 .exe 
PHP 和 其 他 几 种 Web 语言 一 样 ， 其 标记 符 能 够 让 Web 服务 器 识别 PHP 代码 的 开始 和 结束 ， 两 个 标记 
之 间 的 所 有 文本 都 会 被 解释 为 PHP， 而 标记 之 外 的 任何 文本 都 会 被 认为 是 普通 的 HTML， 这 就 是 PHP 标记 
的 作用 。PHP 标记 符 风 格 连 异 ， 按 照 风格 的 不 同 可 划分 为 以 下 4 种 。 
(1) 标准 风格 
<?php 
echo "标准 风格 的 PHP 标记 "; 
人 
这 是 本 书 中 使 用 的 标记 风格 ， 也 是 推荐 读者 使 用 的 标记 风格 。 
(2) 脚本 风格 
<script language="php"> 
echo ' 这 是 脚本 风格 的 标记 '; 
</script> 
在 XHTML 或 者 XML 中 推荐 使 用 这 种 标记 风格 ， 它 符合 XML 语言 规范 的 写法 。 
(3) 简短 风格 
<? 
echo "简短 风格 的 标记 " ; 
?> 
这 种 标记 风格 最 为 简单 ， 输 入 字符 最 少 ， 但 想 要 使 用 它 ， 必 须要 更 改 配置 文件 php.ini。 这 种 标记 风格 
不 推荐 使 用 。 
(4) ASP 风格 
<% 
echo "ASP 风格 的 标记 "; 


%> 


生起 如 果 使 用 简短 风格 “<? ?>” 和 ASP 风格 “<% %>”， 需 要 分 别 在 配置 文件 php.ini 中 做 如 下 设 
置 。 打开 系 统 盘 Windows (以 Windows 2003 Server 操作 系统 为 例 ) 文件 夹 下 的 php.ini 文 件 ， 将 如 下 代 
码 段 中 的 “Off” 改 为 “On”。 更改 后 的 代码 如 下 : 


short_open_tag = On 
asp_tags = On 
保存 修改 后 的 php.ini 文件 ， 然 后 重新 启动 Apache 服务 器 ， 即 可 支持 这 两 种 标记 风格 。 
2.1.2 PHP 注释 应 用 


句 视频 讲解 : 光盘 \TM\Video\ 第 2 章 \PHP 注释 应 用 .exe 
注释 即 代码 的 解释 和 说 明 ， 一 般 添加 到 代码 的 上 方 或 代码 的 尾部 (添加 到 代码 的 尾部 时 ， 代 码 和 注释 
之 间 以 Tab 键 进行 分 隔 ， 以 方便 阅读 程序 ) ， 用 来 说 明代 码 或 函数 的 编写 人 、 用 途 、 时 间 等 。 注 释 不 会 影 
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响 到 程序 的 执行 ， 因 为 注释 部 分 在 执行 时 会 被 解释 器 忽略 。 
PHP 支持 3 种 风格 的 程序 注释 。 
/单行 注释 


<?php 
echo 'PHP 开发 实战 宝典 '; /输出 字符 串 〈 但 单行 标记 后 的 注释 内 容 不 被 输出 ) 


?> 
/#*…#/ 多 行 注释 


echo ' 只 会 看 到 这 句 话 。'; 


?> 


Nn 


Shell 风格 的 注释 
<?php 
echo ' 这 是 Shell 脚本 风格 的 注释 '; # 这 里 的 内 容 是 看 不 到 的 
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人 注意 在 单行 注释 里 的 内 容 不 要 出 现 “?>” 的 标志 , 因为 解释 器 会 认为 PHP 脚本 结束 , 而 去 执行 “?>” 


后 面 的 代码 。 例 如 : 

<?php 
echo ' 这 样 会 出 错 的 ! ! !' # 不 会 看 到 ?> 会 看 到 
?> 


结果 为 :这样 会 出 错 的 111 会 看 到 ?> 


2.2 PHP 的 数据 类 型 


区 4 视频 讲解 : 光盘 \TM\Video\ 第 2 章 \PHP 的 数据 类 型 .exe 

PHP 一 共 支 持 8 种 原始 类 型 ， 包 括 4 种 标量 类 型 ， 即 boolean (布尔 型 ) 、integer ( 整 型 ) 、float/double 
〈 浮 点 型 ) 和 string (字符 串 型 ) ;两 种 复合 类 型 ， 即 array (数组 ) 和 object (对 象 ) ; 两 种 特殊 类 型 ， 即 
resource (资源 ) 与 null。 


2.2.1 标量 数据 类 型 


标量 数据 类 型 是 数据 结构 中 最 基本 的 单元 ， 只 能 存储 一 个 数据 。PHP 中 标量 数据 类 型 包括 4 种 ， 如 
表 2.1 所 示 。 


@ 


第 2 章 PHP 语言 基础 


表 2.1 标量 数据 类 型 


类 型 说 明 
boolean (布尔 型 ) | 这 是 最 简单 的 类 型 。 只 有 两 个 值 ， 真 tmue) 和 假 false) 
string (字符 串 型 ) | 字符 串 就 是 连续 的 字符 序列 ， 可 以 是 计算 机 所 能 表示 的 一 切 字符 的 集合 
integer ( 整 型 ) | 整 型 数据 类 型 只 能 包含 整数 。 这 些 数据 类 型 可 以 是 正 数 或 负数 
float ( 浮 点 型 ) 浮 点 数据 类 型 用 于 存储 数字 ， 和 整 型 不 同 的 是 它 有 小 数位 


1. 布尔 型 (boolean) 

布尔 型 是 PHP 中 较为 常用 的 数据 类 型 之 一 ， 它 保存 一 个 true 值 或 者 false 值 ， 其 中 tue 和 false 是 PHP 
的 内 部 关键 字 。 设 定 一 个 布尔 型 的 变量 ， 只 需 将 true 或 者 false 赋值 给 变量 即 可 。 

例 2.1 通常 布尔 型 变量 都 是 应 用 在 条 件 或 循环 语句 的 表达 式 中 。 下 面 在 过 条 件 语 句 中 判断 变量 $b 中 
的 值 是 否 为 tue， 如 果 为 tue， 则 输出 “变量 $b 为 真 !/”， 否 则 输出 “变量 $b 为 假 !!”。 

实例 代码 如 下 : 〈 实 例 位置 ， 光盘 \TM\Instance\0202.1) 


<?php 
$b = true; // 声 明 一 个 boolean 类 型 变量 ， 赋 初 值 为 true 
if($b == true) // 判 断 变 量 $b 是 否 为 真 
echo 变量 $b 为 真 !; // 如 果 为 真 ， 输 出 “变量 $b 为 真 !” 的 字样 
else 
echo 变量 $b 为 假山 // 如 果 为 假 ， 则 输出 “变量 $b 为 假 !” 的 字样 
?> 


结果 为 ， 变 量 $b 为 真 


pe 
殊 情 况 为 : 0、0.0、"0"、 空 字符 串 (")、 只 声明 没有 赋值 的 数组 等 。 


of 
说 四 美元 符号 $ 是 变量 的 标识 符 ， 所 有 变量 都 是 以 8 开头 的 无论 是 声明 变量 还 是 调用 变量 ， 都 应 使 
用 $。 


2. 字符 串 型 〈string) 

字符 串 是 连续 的 字符 序列 ， 由 数字 、 字 母 和 符号 组 成 。 字 符 串 中 的 每 个 字符 只 占用 一 个 字 节 。 在 PHP 
有 3 种 定义 字符 串 的 方式 ， 分 别 是 单 引号 (') 、 双 引号 〈") 和 界定 符 (<<<) 。 

单 引号 和 双 引 号 是 经 常 被 使 用 的 定义 方式 。 定 义 格式 如 下 : 

<?php 

$a = 字符 串 '; 

?> 


中 


或 
<?php 
$a =" 字 符 串 "; 


?> 

两 者 的 不 同 之 处 在 于 ， 双 引号 中 所 包含 的 变量 会 自动 被 蔡 换 成 实际 数值 ， 而 单 引号 中 包含 的 变量 则 按 
普通 字符 串 输出 。 

例 2.2 下 面 的 实例 分 别 应 用 单 引 号 和 双 引 号 来 输出 同一 个 变量 ， 其 输出 结果 完全 不 同 ， 双 引号 输出 的 
是 变量 的 值 ， 而 单 引号 输出 的 是 字符 串 “$i”。 《实例 位 置 ， 光盘 \TM\Instance\02\2.2) 

实例 代码 如 下 : 


@ 
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<?php 
$i = ' 只 会 看 到 一 遍 ' /声明 一 个 字符 串 变量 
echo "$i"; /用 双 引 号 输出 
echo "<p>"; /输出 段 标记 
echo ‘$i /用 单 引 号 输出 


?> 

运行 结果 如 图 2.1 所 示 。 

两 者 之 间 另 一 处 不 同 点 是 对 转 义 字符 的 使 用 。 使 用 单 引 号 时 ， 
只 要 对 单 引 号 〈') 进行 转 义 即 可 ， 但 使 用 双 引 号 〈") 时 ， 还 要 注 
意 “"”、“$” 等 字符 的 使 用 。 这 些 特 殊 字符 都 要 通过 转 义 符 (\) 


来 显示 。 常 用 的 转 义 字符 如 表 2.2 所 示 。 2 
表 2.2 转 义 字 符 
转 义 字符 输 出 
nm 换行 (LF 或 ASCII 字符 0x0A (10) ) 
Yr 回 车 (CR 或 ASCI 字符 0x0D (13) ) 
Vt 水 平 制 表 符 (HT 或 ASCII 字符 0x09 (9) ) 
\ 反 斜 杠 
\$ 美元 符号 
VY 单 引号 
六 双 引 号 
\[0-7]{1.3 此 正则 表达 式 序列 匹配 一 个 用 八进制 符号 表示 的 字符 ， 如 M467 
\x[0-9A-Fa-f]{1.2 此 正则 表达 式 序列 匹配 一 个 用 十 六 进 制 符号 表示 的 字符 ， 如 wx9f 


Wn 和 \ 在 Windows 系统 中 没有 什么 区 别 ， 都 可 以 当 作 回 车 符 。 但 在 Linux 系统 中 则 是 两 种 效果 ，wn 表 
示 换 到 下 一 行 ， 却 不 会 回 到 行 首 ， 而 Y 表示 光标 回 到 行 首 ， 但 仍然 在 本 行 。 如 果 读者 使 用 Linux 操作 系统 ， 
可 以 尝试 一 下 。 


\ 


a. 
SA 在 定义 简单 的 字符 串 时 ， 使 用 单 引 号 是 一 个 更 加 合适 的 处 理 方式 。 如 果 使 用 双 引 号 ，PHP 将 
花费 一 些 时 间 来 处 理 字符 串 的 转 义 和 变量 的 解析 。 因 此 ， 在 定义 字符 串 时 ， 如 果 没 有 特别 的 要 求 ， 应 尽 
量 使 用 单 引号 。 


界定 符 (<<<) 是 从 PHP 4.0 开始 支持 的 。 在 使 用 时 后 接 一 个 标识 符 ， 然 后 是 字符 串 ， 最 后 是 同样 的 标 
识 符 结 束 字符 串 。 界 定 符 的 格式 如 下 : 

$string = <<< str 

要 输出 的 字符 串 。 

str 


其 中 str 为 指定 的 标识 符 。 
例 2.3 下 面 使 用 界定 符 输出 变量 中 的 值 ， 可 以 看 到 ， 它 和 双 引 号 没什么 区 别 ， 包 含 的 变量 也 被 蔡 换 成 
实际 数值 。〈 实 例 位 置 : 光盘 \TM\Instance\02\2.3) 
实例 代码 如 下 : 
<?php 
$i = ' 显 示 该 行内 容 ' // 声 明 变 量 $i 


@ 
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echo <<<std // 界 定 符 开始 
这 和 双 引 号 没有 什么 区 别 ，\$i 同样 可 以 被 输出 出 来 。<p> // 输 出 字符 串 
$i 的 内 容 为 : $i // 输 出 变量 $i 
std; /界定 符 结束 
人 
运行 结果 如 图 2.2 所 示 。 


Wy a 
生 错 误 , 例 2.3 中 的 注释 部 分 在 练习 时 一 定 不 要 输入 ,否则 将 出 现 “Parse error: parse error unexpected TT_SL 
in E:\AppServWwww\tm\02\2.3\index.php on line…” 的 错误 提示 。 


3. 整 型 (integer) 


整 型 数据 类 型 只 能 包含 整数 。 在 32 位 的 操作 系统 中 ， 有 效 的 范围 是 -2147483648 一 +2147483647。 整 型 
数 可 以 用 十 进 制 、 八 进 制 和 十 六 进 制 来 表示 。 如 果 用 八进制 ， 数 字 前 面 必须 加 0， 如 果 用 十 六 进 制 ， 则 需要 
加 0x。 


et 


例 2.4 本 例 分 别 输出 八进制 、 十 进 制 和 十 六 进 制 的 结果 。 《实例 位 置 : 光盘 \TMNInstance\02\2.4) 
实例 代码 如 下 : 


<?php 
Sstr1 = 1234567890; /声明 一 个 十 进 制 的 整数 
$str2 = 0x1234567890; /声明 一 个 十 六 进 制 的 整数 
$str3 = 01234567890; /声明 一 个 八进制 的 整数 
$str4 = 01234567; // 声 明 另 一 个 八进制 的 整数 
echo 数字 1234567890 不 同 进 制 的 输出 结果 : <p>'; 
echo '10 进 制 的 结果 是 : '.$str1.'<br>'; // 输 出 十 进 制 整数 
echo '16 进 制 的 结果 是 :'.$str2.'<br>"; // 输 出 十 六 进 制 整数 
echo '8 进 制 的 结果 是 : “ 
if($str3 == $str4){ /判断 $str3 和 S$str4 的 关系 
echo '$str3 = $str4 = '.$str3; // 如 果 相等 ， 输 出 变量 值 
Jelse{ 
echo '$str3 != str4'; // 如 果 不 相等 ， 输 出 “$str3 = $str4” 
} 
?> 
运行 结果 如 图 2.3 所 示 。 
大 不必 过 4 肘 = 给 果 - Windows Incernet Explorer 
RR Wndow emer Erplorer ts ed |e sochoscri mmo) [|x |[P sems p ~ 
Sor er' tp/ocalhast32/ THD p= EE = 
ND MRE SE) WA IRMD fm wa i 从 ~” 园 - 马 存 "~ 
渡 0 宪 。 着 信用 界定 全 定义 字符 和合 " 目 "口才 * - 
过 和 引号 用 有 什么 区 别 。$; 周 竹本 以 裕 加 出 出 来。 国 | 上 
5: 的 内 容 为 + 导 示 该 行内 雁 
避 不 地 Intaret | 人 FP 守 克 倚 用 和 > 大 100% ~ 所 10D% 
图 2.2 使 用 界定 符 定义 字符 串 图 2.3 不 同 进 制 的 输出 结果 


《5 注意 如 果 给 定 的 数值 超出 了 int 型 所 能 表示 的 最 大 范围 ,将 会 被 当 作 float 型 处 理 , 这 种 情况 称 为 整 
数 溢出 。 同 样 ， 如 果 表 达 式 的 最 后 运算 结果 超出 了 int 型 的 范围 ， 也 会 返回 float 型 。 
© 
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4. 浮 点 型 (float) 

浮 点 数据 类 型 可 以 用 来 存储 数字 ， 也 可 以 保存 小 数 。 它 提供 的 精度 比 整数 大 得 多 。 在 32 位 的 操作 系统 
中 ， 有 效 的 范围 是 1.7E-308 一 1.7E+308。 在 PHP 4 以 前 的 版 本 中 ， 浮 点 型 的 标识 为 double， 也 叫做 双 精 度 
浮 点 数 ， 两 者 没有 区 别 。 

浮 点 型 数据 默认 有 两 种 书写 格式 ， 一 种 是 标准 格式 : 

3.1415 

-35.8 

还 有 一 种 是 科学 计数 法 格式 : 

3.58E1 

849.72E-3 

例 2.5 本 例 中 输出 圆周 率 的 近似 值 。 用 3 种 书写 方法 : 圆周 率 函数 、 传 统 书写 格式 和 科学 记 数 法 ， 最 
后 显示 在 页 面 上 的 效果 都 一 样 。〔 实 例 位 置 ， 光盘 \TM\Instance\02\2.5) 


实例 代码 如 下 : 

<?php 

echo ' 圆 周 率 的 3 种 书写 方法 : <p>'; 

echo "第 一 种 : pi() ='… pi() .'<p>"; // 调 用 pi() 函 数 输出 圆周 率 
echo ' 第 二 种 : 3.14159265359 = '. 3.14159265359 .'<p>"; // 传 统 书写 格式 的 浮 点 数 
echo ' 第 三 种 : 314159265359E-11 =' 314159265359E-11 .'<p>"; // 科 学 计数 法 格式 的 浮 点 数 
2 


运行 结果 如 图 2.4 所 示 。 


和 6 让 浮 点 型 的 数值 只 是 一 个 近似 值 ， 所 以 要 尽量 避免 
浮 点 型 数值 之 间 比 较 大 小 , 因为 最 后 的 结果 往往 是 不 准确 的 。 


2.2.2 ”复合 数据 类 型 


复合 数据 类 型 包括 两 种 ， 即 数组 和 对 象 ， 如 表 2.3 所 示 。 
表 2.3 复合 数据 类 型 


图 2.4 输出 浮 点 类 型 


一 组 类 型 相同 的 变量 的 集合 
object (对象) 对 象 是 类 的 实例 ， 使 用 new 命令 来 创建 


1. 数组 (array) 

数组 是 一 组 数据 的 集合 ， 它 把 一 系列 数据 组 织 起 来 ， 形 成 一 个 可 操作 的 整体 。 数 组 中 可 以 包括 很 多 数 
据 ， 如 标量 数据 、 数 组 、 对 象 、 资 源 以 及 PHP 中 支持 的 其 他 语法 结构 等 。 

数组 中 的 每 个 数据 称 为 一 个 元 素 ， 元 素 包 括 索引 〈 键 名 ) 和 值 两 个 部 分 。 元 素 的 索引 可 以 由 数字 或 字 
符 串 组 成 ， 元 素 的 值 可 以 是 多 种 数据 类 型 。 定 义 数组 的 语法 格式 如 下 : 

$array = (value1, value2 '......) 

或 

$array[key] = value' 

或 

$array = array(key1 => value1, key2 => value2...... ) 
其 中 ， 参 数 key 是 数组 元 素 的 下 标 ，value 是 数组 下 标 所 对 应 的 元 素 。 以 下 几 种 都 是 正确 的 格式 : 
$arr1 = array('This','is','a','example’); 


@ 
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$arr2 = array(0 => 'php', 1=>'is', the' => 'the'，'str => 'best "); 

$arr3[0] = tmpname'; 

声明 数组 后 ， 数 组 中 的 元 素 个 数 还 可 以 自由 更 改 。 只 要 给 数组 赋值 ， 数 组 就 会 自动 增加 长 度 。 在 第 7 
章 中 会 详细 介绍 数组 的 使 用 、 取 值 以 及 数组 的 相关 函数 。 

2. 对 象 (object) 

编程 语言 所 应 用 到 的 方法 有 两 种 : 面向 过 程 和 面向 对 象 。 在 PHP 中 ， 用 户 可 以 自由 使 用 这 两 种 方法 。 
在 第 8 章 中 将 对 面向 对 象 的 技术 进行 详细 讲解 。 


2.2.3 ”特殊 数据 类 型 


特殊 数据 类 型 包括 资源 和 空 值 两 种 ， 如 表 2.4 所 示 。 
表 2.4 特殊 数据 类 型 
说 明 

资源 是 一 种 特殊 变量 ， 又 叫做 句柄 ,保存 到 外 部 资源 的 一 个 引用 。 资 源 是 通过 专门 的 函数 来 建立 
和 使 用 的 
null( 空 值 ) 特殊 的 值 ， 表 示 变 量 没有 值 ， 唯 一 的 值 就 是 null 

1. 资源 (resource) 

资源 类 型 是 PHP 4 引进 的 。 关 于 资源 的 类 型 ， 可 以 参考 PHP 手册 后 面 的 附录 ， 里 面 有 详细 的 介绍 和 
说 明 。 

在 使 用 资源 时 ， 系 统 会 自动 启用 垃圾 回收 机 制 ， 释 放 不 再 使 用 的 资源 ， 避 免 内 存 消耗 殉 尽 。 因 此 ， 资 
源 很 少 需要 手工 释放 。 

2. 空 值 (null) 

室 值 ， 顾 名 思 义 ， 表 示 没 有 为 该 变量 设置 任何 值 ， 另 外 ， 空 值 Cnull) 不 区 分 大 小 写 ，null 和 NULL 效 
果 是 一 样 的 。 被 赋予 空 值 的 情况 有 以 下 3 种 : 还 没有 赋 任 何 值 、 被 赋值 null、 被 unsetO 函 数 处 理 过 的 变量 。 

例 2.6 下 面 来 看 一 个 具体 实例 。 字 符 串 stringl 被 赋值 为 null，string2 根本 没有 声明 和 赋值 ， 所 以 也 输 
出 null， 最 后 的 string3 虽然 被 赋予 了 初 值 ， 但 被 unset0 函 数 处 理 后 ， 也 变 为 null 型 。unset0 函 数 的 作用 就 是 
从 内 存 中 删除 变量 。〈 实 例 位 置 : 光盘 \TMNInstance\02\2.6) 

实例 代码 如 下 : 


类 型 


resource (资源) 


<?php 

echo "变量 (\$string1) 直 接 赋值 为 null: “; 

$string1 = null; /变量 $string1 被 赋 空 值 
$string3 = "str"; // 变 量 $string3 被 赋值 str 
if(is_null($string1)) /| 判断 $string1 是 否 为 空 


echo "string1 = null"; 

echo "<p> 变 量 (\$string2) 未 被 赋值 : "; 

if(is_null($string2)) /| 判断 $string2 是 否 为 空 
echo "string2 = null"; 

echo "<p> 被 unset() 函 数 处 理 过 的 变量 (\$string3):“; 

unset($string3); /释放 $string3 

if(is_null($string3)) // 判 断 $string3 是 否 为 空 
echo "string3 = null"; 

> 
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运行 结果 如 图 2.5 所 示 。 gE ~ 
sa/ z 
NE 说明 es 
is_null0 函 数 是 判断 变量 是 否 为 null， 该 函数 返回 Ce = 
一 个 boolean 型 ,如 果 变量 为 null, 则 返回 tmue, 否则 返回 false。 ww 上 
UnsetO 函 数 用 来 销毁 指定 的 变量 。 Rn 


图 2.5 被 赋值 为 null 的 几 种 情况 
人 注意 pmp fi unset0 函 数 就 不 再 有 返回 值 ， 所 以 不 要 试图 获取 或 输出 unsetO。 


2.2.4 ”转换 数据 类 型 


虽然 PHP 是 弱 类 型 语言 , 但 有 时 仍然 需要 用 到 类 型 转换 。PHP 中 的 类 型 转换 和 C 语言 一 样 , 非常 简单 ， 
只 需 在 变量 前 加 上 用 括号 括 起 来 的 类 型 名 称 即 可 。 人 允许 转换 的 类 型 如 表 2.5 所 示 。 


表 2.5 ”类 型 强制 转换 


(boolean)$num、(boolean)$str 
(string)$boo、(string)$flo 


(integer)$boo、(integer)$str 
(float)$str、(float)$str 


(object)$str 


和 0 站 在 进行 类 型 转换 的 过 程 中 应 该 注意 以 下 内 容 : 转换 成 boolean 型 时 ，null、0 和 未 赋值 的 变量 或 
数组 会 被 转换 为 false， 其 他 的 为 真 ; 转换 成 整 型 时 ， 布 尔 型 的 false 转换 为 0，true 转换 为 1， 浮 点 型 的 
小 数 部 分 被 舍 去 ， 字 符 型 如 果 以 数字 开头 就 截取 到 非 数 字 位 ， 否 则 输出 0。 


类 型 转换 还 可 以 通过 settype0 函 数 来 完成 ， 该 函数 可 以 将 指定 的 变量 转换 成 指定 的 数据 类 型 。 
bool settype ( mixed var, string type ) 
参数 var 为 指定 的 变量 ， 参 数 type 为 指定 的 类 型 ， 有 7 个 可 选 值 ， 即 boolean、float、integer、array、 


null、object 和 string。 如 果 转 换 成 功 则 返回 tue， 否 则 返回 false。 

当 字 符 串 转换 为 整 型 或 浮 点 型 时 ， 如 果 字 符 串 是 以 数字 开头 的 ， 就 会 先 把 数字 部 分 转换 为 整 型 ， 再 舍 
去 后 面 的 字符 串 ， 如 果 数字 中 含有 小 数 点 ， 则 会 取 到 小 数 点 前 一 位 。 

例 2.7 ”本 实例 将 使 用 上 面 的 两 种 方法 将 指定 的 字符 串 进行 类 型 转换 ， 比 较 两 种 方法 之 间 的 不 同 。 ( 实 
例 位 置 ， 光盘 \TMNVInstance\02\2.7) 

实例 代码 如 下 : 


<?php 

$num = '2.1415926rr'; // 声 明 一 个 字符 串 变量 
echo ' 使 用 (integen) 操 作 符 转换 变量 Snum 类 型 :'; 

echo (integer)$num: /使 用 integer 转换 类 型 
echo '<p>"; 

echo ' 输 出 变量 Snum 的 值 : '.$num; /| 输出 原始 变量 Snum 


@ 


echo '<p>"; 

echo ' 使 用 settype 函数 转换 变量 Snum 类 型 : “; 
echo settype($num,'integer); 

echo '<p>"; 


echo 输出 变量 Snum 的 值 : -$num; 
2> 


运行 结果 如 图 2.6 所 示 。 
可 以 看 到 , 使 


回 的 是 1， 
可 根据 情况 自行 选择 转换 方式 。 


2.2.5 ”检测 数据 类 型 


PHP 还 内 置 了 检测 数据 类 型 的 系列 函数 ， 


integer 操作 符 能 直接 输出 转换 后 的 变量 
类 型 。 并 且 原 变量 不 发 生 任何 变化 。 而 使 用 settype0 函 数 返 
也 就 是 tue， 而 原 变量 被 改变 了 。 在 实际 应 用 中 ， 
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/使 用 settype 函数 转换 类 型 
// 输 出 原始 变量 $num 


图 2.6 类 型 转换 
可 以 对 不 同类 型 的 数据 进行 检测 ， 判 断 其 是 否 属于 某 个 类 型 ， 


如 果 符合 则 返回 rue， 否则 返回 false。 检 测 数 据 类 型 的 函数 如 表 2.6 所 示 。 
表 2.6 检测 数据 类 型 
函数 检测 类 型 举例 
is bool 检查 变量 是 否 是 布尔 类 型 is bool(true)、 is book(false) 
is_string 检查 变量 是 否 是 字符 串 类 型 is_string(string)、is_string(1234) 
is_floatis_ double 检查 变量 是 否 为 浮 点 类 型 is float(3.1415)、 is float(3.1415)) 
is integer/is int 检查 变量 是 否 为 整数 is integer(34)、is integer('34") 
is null 检查 变量 是 否 为 null is nullnull 
is_arra 检查 变量 是 否 为 数组 类 型 is_array($arr) 
is_object 检查 变量 是 否 是 一 个 对 象 类 型 is_object($obj) 
is numeric 检查 变量 是 否 为 数字 或 由 数字 组 成 的 字符 串 is numeric(S)、is_ numeric(bccd110') 


例 2.8 ”由 于 检测 数据 类 型 的 函数 的 功 外 
的 数据 是 否 是 数字 ， 从 而 了 解 并 掌握 is 系列 函数 的 用 法 。 


实例 代码 如 下 : 
<?php 
$boo = "043112345678"; 
if(is_numeric($boo)) 
echo "Yes,the \$boo a phone number: $boo!"; 
else 
echo "Sorry, This is an error!™; 
Wx 


结果 为 : Yes,the $boo a phone number: 043112345678。 


2.3 PHP 的 常 


涛 和 用 法 都 是 相同 的 ， 


下 面 使 用 is_numeric() 函 数 来 检测 变量 中 
(实例 位 置 ， 光盘 \TM\Instance\02\2.8) 


// 声 明 一 个 全 由 数字 组 成 的 字符 串 变 量 
// 判 断 该 变量 是 否 由 数字 组 成 
// 如 果 是 ， 输 出 该 变量 


/否则 输出 错误 语句 


二 


耐量 应 


应 用 


铭 中 视频 讲解 : 光盘 \IM\Video\ 第 2 章 \PHP 的 常量 应 用 .exe 


常量 可 以 理解 为 用 于 存储 不 经 常 改变 的 数据 信息 的 量 。 


常量 的 值 被 确定 后 ， 在 程序 的 整个 执行 期 间 内 ， 


人 @ 
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这 个 值 都 有 效 ， 并 且 不 可 再 次 对 该 常量 进行 赋值 。 本 节 介绍 PHP 的 常量 ， 包 括 常量 的 声明 和 使 用 ， 以 及 预 
定义 常量 。 


2.3.1 声明 和 使 用 常量 


一 个 常量 由 英文 字母 、 下 划 线 和 数字 组 成 ， 但 数字 不 能 作为 首 字母 出 现 。 

在 PHP 中 使 用 define0 函 数 来 定义 常量 ， 该 函数 的 语法 格式 为 : 

define(string constant_name,mixed value,case_sensitive=true) 

constant name: 必 选 参数 ， 常 量 名 称 ， 即 标识 符 。 

value: 必 选 参数 ， 常 量 的 值 。 

加 ”case_sensitive: 可 选 参数 , 指定 是 否 大 小 写 敏 感 ， 设 定 为 tme 表示 不 敏感 。 默 认 状态 表示 大 小 写 敏 感 。 

获取 常量 值 有 两 种 方法 : 一 种 是 使 用 常量 名 直接 获取 值 ， 另 一 种 是 使 用 constantO 函 数 ， 这 和 直接 使 用 
常量 名 输出 的 效果 是 一 样 的 。 但 函数 可 以 动态 输出 不 同 的 常量 ， 在 使 用 上 要 灵活 、 方 便 得 多 。 函 数 的 语法 
格式 为 : 

mixed constant(string const_name) 

参数 const_name 为 要 获取 常量 的 名 称 ， 也 可 为 存储 常量 名 的 变量 。 如 果 成 功 则 返回 常量 的 值 ， 失 败 则 
提示 错误 信息 “常量 没有 被 定义 ”。 

要 判断 一 个 常量 是 否 已 经 定义 ， 可 以 使 用 defined0 函 数 。 函 数 的 语法 格式 为 

bool defined(string constant_name); 

参数 constant_name 为 要 获取 常量 的 名 称 ， 成 功 则 返回 twe， 否 则 返回 false。 

例 2.9 为 了 便于 读者 更 好 地 理解 如 何 定义 常量 ， 这 里 给 出 一 个 定义 常量 的 实例 。 在 这 个 实例 中 应 用 上 
述 的 3 个 函数 : define0 函 数 、constant0 函 数 和 defined0 〇 函数 ,通过 define0 〇 函数 来 定义 一 个 常量 ,使 用 constantO) 
函数 来 动态 获取 常量 的 值 , 应 用 defined0 函 数 来 判断 常量 是 否 被 定义 。( 实 例 位 置 : 光盘 \TM\Instance\02\2.9) 


代码 如 下 : 

<?php 

define ("MESSAGE"," 阴 日 科技 "); // 定 义 常量 ， 并 设置 大 小 写 敏感 

echo MESSAGE."<BR>"; // 输 出 常量 MESSAGE 

echo Message."<BR>"; /输出 "Message"， 表 示 没有 该 常量 

define ("COUNT"," 阴 日 科技 ， 让 您 尽 在 其 中 ",true); 

echo COUNT."<BR>"; // 输 出 常量 COUNT 

echo Count."<BR>"; /| 输出 常量 COUNT， 因 为 设 定 大 小 写 不 敏感 

S$name = "count"; 

echo constant ($name)."<BR>"; // 输 出 常量 COUNT 

if (defined ("MESSAGE")X // 如 果 定 义 返回 true， 使 用 echo 输出 显示 信息 
echo "明日 科技 是 一 家 知名 企业 的 软件 公司 ! "; 

加 

运行 结果 如 图 2.7 所 示 。 


2.3.2 ”预定 义 常量 


PHP 中 可 以 使 用 预定 义 常量 获取 PHP 中 的 信息 。 常 
用 的 预定 义 常 量 如 表 2.7 所 示 。 


@ 


ET 而 ”00 ~ 


图 2.7 通过 函数 对 常量 进行 定义 、 获 取 和 判断 


表 2.7 PHP 的 预定 义 常 量 


常 量 名 功 能 
FILE 默认 常量 ，PHP 程序 文件 名 
LINE 默认 常量 ，PHP 程序 行 数 
PHP VERSION 内 建 常量 ，PHP 程序 的 版 本 ， 如 3.0.8_dev 
PHP OS 内 建 常量 ， 执 行 PHP 解析 器 的 操作 系统 名 称 ， 如 Windows 
TRUE 这 个 常量 是 一 个 真 值 (true) 
FALSE 这 个 常量 是 一 个 假 值 (false) 
NULL 一 个 null 值 
E ERROR 这 个 常量 指 到 最 近 的 错误 处 
E WARNING 这 个 常量 指 到 最 近 的 警告 处 
E PARSE 这 个 常量 指 解 析 语 法 有 潜在 问题 处 
E NOTICR. 这 个 常量 为 发 生 不 寻常 的 提示 但 不 一 定 是 错误 处 


人 es 注 意 “FILE ”和 “ LINE ”中 的 “ ”是 两 条 下 划 线 ,而 不 是 一 条 ““”。 


4 
5 明 表 中 以 “EE ”开头 的 预定 义 常量 ， 是 PHP 的 错误 调试 部 分 。 如 果 想 详细 了 解 相关 内 容 ， 可 参 
考 error_reporting() 函 数 。 


例 2.10 预定 义 常量 与 用 户 自 定义 常量 在 使 用 上 没什么 差别 。 下 面 应 用 预定 义 常 量 来 输出 PHP 中 的 信 


息 。 (实例 位 置 ， 光盘 \TM\Instance\02\2.10) 
实例 代码 如 下 : 
<?php 
echo "当前 文件 路 径 : ".__FILE_; // 输 出“_FILE_” 常 量 
echo "<br> 当 前 行 数 : "LINE_; // 输 出 “_LINE_” 常 量 
echo "<br> 当 前 PHP 版 本 信息 : "PHP_VERSION; /| 输出 PHP 版 本 信息 
echo "<br> 当前 操作 系统 : "PHP_OS ; // 输 出 系统 信息 
2 


运行 结果 如 图 2.8 所 示 。 


BO ERE Wndove Imremet Freer 


图 2.8 ”应 用 PHP 预定 义 常量 输出 信息 


po 


2.4 PHP 的 变量 应 用 


锋 4 视频 讲解 : 光盘 \TM\Video\ 第 2 章 \PHP 的 变量 应 用 .exe 
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变量 是 指 在 程序 执行 过 程 中 数字 可 以 变化 的 量 。 变 量 通 过 一 个 名 字 (变量 名 ) 来 标识 。 系 统 为 程序 中 
的 每 一 个 变量 分 配 一 个 存储 单元 ， 变 量 名 实质 上 就 是 计算 机 内 存单 元 的 命名 。 因 此 ， 借 助 变量 名 即 可 访问 
内 存 中 的 数据 。 


2.4.1 变量 声明 及 使 用 


和 很 多 语言 不 同 ， 在 PHP 中 使 用 变量 前 不 需要 声明 变量 (PHP 4.0 之 前 需要 声明 变量 ) ， 只 需 为 变量 赋 
值 即 可 。PHP 中 的 变量 名 称 用 $ 和 标识 符 表示 ， 变 量 名 是 区 分 大 小 写 的 。 
PHP 中 的 变量 名 称 遵 循 以 下 约定 。 
在 PHP 中 的 变量 名 是 区 分 大 小 写 的 。 
变量 名 必须 是 以 美元 符号 〈$) 开始 。 
变量 名 开头 可 以 以 下 划 线 开始 。 
变量 名 不 能 以 数字 字符 开头 。 
变量 名 可 以 包含 一 些 扩展 字符 〈 如 重音 拉丁 字母 ) ， 但 不 能 包含 非法 扩展 字符 〈 如 汉字 字符 和 汉 
字 字 母 ) 。 


加 


办 办 办 轨 


有 


[3 技巧 声明 的 变量 不 可 以 与 已 有 的 变量 重 名 , 否则 将 引起 冲突 。 变量 的 名 称 应 采用 能 反映 变量 含义 的 
名 称 ， 以 利于 提高 程序 的 可 读 性 。 只 要 能 明确 反映 变量 的 含义 ,可 以 使 用 英文 单词 、 单词 缩写 、 拼音 ( 尽 
量 使 用 英文 单词 )， 如 $book_name、S$user age、$shop_price 等 ， 必 要 时 ， 也 可 以 将 变量 的 类 型 包含 在 变 
量 名 中 ， 如 $book id int， 这 样 可 以 直接 根据 变量 名 称 了 解 变量 的 类 型 。 


在 程序 中 使 用 变量 前 ， 需 要 为 变量 赋值 。PHP 中 变量 的 定义 非常 简单 、 灵 活 。 在 定义 变量 时 ， 不 需要 
指定 变量 的 类 型 ，PHP 自动 根据 对 变量 的 赋值 决定 其 类 型 。 变 量 的 赋值 是 通过 使 用 赋值 运算 符 (=) 实现 的 。 
在 定义 变量 时 也 可 以 直接 为 变量 赋值 ， 此 时 称 之 为 变量 的 初始 化 。 

例 2.11 下 面 的 代码 定义 了 一 个 整 型 变量 n_sum, 将 其 赋值 为 100; 定义 一 个 布尔 型 变量 ; 定义 一 个 空 
字符 串 。 (实例 位 置 ， 光盘 \TMNVInstance\02\2.11) 


<?php 
$n_sum = 100; // 定 义 一 个 整 型 变量 ， 并 进行 初始 化 
S$str1=false; // 定 义 一 个 布尔 型 变量 ， 并 进行 初始 化 
$str2=" "; // 定 义 一 个 空 字符 串 


> 


和 ot 在 定义 变量 时 ， 要 养 成 良好 的 编程 习惯 ， 在 定义 变量 前 要 对 其 定义 初始 值 。 如 果 在 定义 变量 时 
没有 指定 变量 的 初始 值 ， 那 么 在 使 用 变量 时 ，PHP 会 根据 变量 在 语句 中 所 处 的 位 置 确定 其 类 型 ， 并 采用 
该 类 型 的 默认 值 。 字 符 串 的 初始 值 为 空 值 ; 整 型 的 初始 值 为 0; 布尔 型 的 初始 值 为 false。 


对 变量 赋值 时 ， 要 遵循 变量 命名 规则 ， 如 下 面 的 变量 命名 是 合法 的 。 
<?php 
$helpsoft=" 明 日 科技 有 限 公司 "; 
$_book= "软件 公司 “; 
?> 
下 面 的 变量 命名 是 非法 的 。 
<?php 


局 
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$5_str=" 阴 日 科技 "; /变量 名 不 能 以 数字 字符 开头 
S$@zts = "mrkj"; /变量 名 不 能 以 其 他 字符 开头 


?> 


二 PHP 中 的 变量 名 称 区 分 大 小 写 ， 而 函数 名 称 不 区 分 大 小 写 。 


除了 直接 赋值 外 ， 还 有 两 种 方式 来 给 变量 声明 或 赋值 。 一 种 是 变量 间 的 赋值 。 结 果 为 : 明日 科技 有 限 
公司 。 另 一 种 是 引用 赋值 。 从 PHP 4.0 开始 ，PHP 引入 了 “引用 赋值 ”的 概念 。 引 用 赋值 是 指 用 不 同 的 名 字 
访问 同一 个 变量 内 容 ， 当 改变 其 中 一 个 变量 的 值 时 ， 另 一 个 也 跟着 发 生变 化 。 使 用 & 符 号 来 表示 引用 。 

例 2.12 ”变量 间 的 赋值 是 指 赋值 后 ， 两 个 变量 使 用 各 自 的 内 存 ， 互 不 干扰 。《〈 实 例 位 置 : 光盘 \TM 
Instance\02\2.12) 


实例 代码 如 下 : 

<?php 

$str1 = "明日 科技 有 限 公司 "; /声明 变量 $str1 

$str2 = $str1; /使 用 $str1 来 初始 化 $str2 
$str1 = "我 喜欢 学 PHP"; /改变 变量 $str1 的 值 
echo $str2; // 输 出 变量 $str2 的 值 

?> 


例 2.13 本 例 中 变量 $str2 是 变量 $str 的 引用 ， 当 给 变量 $str 赋值 后 ，$str2 的 值 也 会 跟着 发 生变 化 。( 实 
例 位 置 ， 光盘 \TMNInstance\02\2.13) 


实例 代码 如 下 : 

<?php 

$str = "是 一 家 知名 的 软件 公司 "; /声明 变量 $str 

$str2 = & $str; // 使 用 引用 赋值 ， 这 是 $str2 已 经 赋值 成 为 “是 一 家 知名 的 软件 公司 ” 
$str = "明日 科技 : $str"; // 重 新 给 $str 赋值 

echo $str2; /输出 变量 $str2 

echo "<p>"; /输出 换行 标记 

echo S$str; /输出 变量 $str 

?> 

运行 结果 如 图 2.9 所 示 。 nn 


引用 变量 并 不 是 复制 一 个 变量 给 另 一 个 变量 , 而 是 将 两 个 
变量 指向 同一 个 内 容 , 可 以 理解 为 将 同一 个 变量 的 地 址 传递 给 
另 一 个 变量 。 引 用 后 ， 两 个 变量 完全 相同 。 当 对 其 中 的 任意 一 
个 变量 的 内 容 进 行 更 改 时 ， 另 一 个 变量 的 内 容 也 会 随 之 更 改 。 

引用 和 复制 的 区 别 在 于 : 复制 是 将 原 变量 内 容 复 制 下 来 ， 
开辟 一 个 新 的 内 存 空间 来 保存 。 而 引用 则 是 给 变量 的 内 容 再 命 图 2.9 引用 赋值 
一 个 名 字 。 也 可 以 这 样 理 解 ， 一 些 论坛 的 版 主 、 博 客 的 博 主 ， 登 录 网 站 时 发 表 帖子 或 文章 时 一 般 不 会 留 真 
名 ， 而 是 用 笔名 ， 这 个 笔名 就 可 以 看 作 是 一 个 引用 ， 是 其 身份 的 代表 。 


2.4.2 ”变量 作用 域 
变量 在 使 用 时 ， 要 符合 变量 的 定义 规则 。 变 量 必须 在 有 效 范围 内 使 用 ， 如 果 变 量 超出 有 效 范围 ， 变 量 


也 就 失去 其 意义 了 。 按 作用 域 可 以 将 变量 分 为 全 局 变量 、 局 部 变量 和 静态 变量 。 变 量 作用 域 的 说 明 如 表 2.8 
所 示 。 
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表 2.8 变量 作用 域 
说 明 

被 定义 在 所 有 函数 以 外 的 变量 ， 其 作用 域 是 整个 PHP 文件 ， 但 是 在 用 户 自 定义 函数 内 部 是 不 可 用 的 。 
想 在 用 户 自 定义 函数 内 部 使 用 全 局 变量 ， 要 使 用 关键 字 global 声明 ， 或 者 通过 使 用 全 局 数组 $globals 进 
行 访问 
在 函数 内 部 定义 的 变量 ， 只 限于 在 函数 内 部 使 用 ， 在 函数 外 部 不 能 被 使 用 
能 够 在 函数 调用 结束 后 仍 保留 变量 值 ， 当 再 次 回 到 其 作用 域 时 ， 又 可 以 继续 使 用 原来 的 值 。 而 一 般 变量 
是 在 函数 调用 结束 后 ， 其 存储 的 数据 值 将 被 清除 ， 所 占 的 内 存 空间 被 释放 。 使 用 静态 变量 时 ， 先 要 用 关 
键 字 static 来 声明 变量 ， 需 要 把 关键 字 static 放 在 要 定义 的 变量 之 前 


在 函数 的 内 部 定义 的 变量 ， 其 作用 域 是 所 在 函数 。 如 果 在 函数 外 赋值 ， 将 被 认为 是 完全 不 同 的 另 一 个 
变量 。 在 退出 声明 变量 的 函数 时 ， 该 变量 及 相应 的 值 就 会 被 撤销 。 

例 2.14 下 面 在 自 定义 函数 中 应 用 全 局 变量 与 局 部 变量 进行 对 比 。 在 本 实例 中 定义 两 个 全 局 变量 $zy 
和 $zyy， 在 用 户 自 定义 函数 Ixt0 中 ， 如 果 想 要 在 第 6 行 和 第 8 行 调用 它们 ， 而 程序 输出 的 结果 是 “明日 科技 
有 限 公 司 ”， 因 为 在 第 7 行 用 关键 字 global 声明 了 全 局 变量 $zyy。 而 第 5 行 由 于 定义 了 局 部 变量 ， 因 此 输出 
的 结果 为 “明日 科技 ”, 其 中 第 5 行 的 $zy 和 第 2 行 的 Szy 没 有 任何 关系 .( 实 例 位 置 :光盘 \TM\Instance\02\2.14) 


作 用 域 


全 局 变量 


局 部 变量 


静态 变量 


实例 代码 如 下 : 
<?php 
$zy = "你 好 " ; 
$zyy = "有 限 公 司 " ; 
function Ixt (X{ 
$zy=" 明 日 科技 "; // 定 义 局 部 变量 
echo $zy; /zy 输出 的 是 局 部 变量 的 内 容 ， 而 并 非 全 局 
global $zyy ; /应 用 关键 字 global 在 函数 内 部 定义 全 局 变量 
echo $zyy."<br>" ; // 此 处 调用 $zyy 
} 
Ixt () ， 
Ne 


运行 结果 : 明日 科技 有 限 公司 。 


榴 s 沪 因为 默认 情况 下 全 局 变量 和 局 部 变量 的 作用 域 是 不 相交 的 , 所以, 在 函数 内 部 可 以 定义 与 全 局 
变量 同名 的 变量 。 全 局 变量 可 以 在 程序 中 的 任何 地 方 访问 。 但 是 在 用 户 自 定义 函数 内 部 是 不 可 用 的 ， 想 
在 用 户 自 定义 函数 内 部 使 用 全 局 变量 ， 要 使 用 关键 字 global 声明 。 


静态 变量 在 函数 内 部 定义 ， 只 局 限于 函数 内 部 使 用 ， 但 却 具有 和 程序 文件 相同 的 生命 周期 。 也 就 是 说 ， 
静态 变量 一 旦 被 定义 ， 在 当前 程序 文件 结束 之 前 一 直 存 在 。 

静态 变量 通过 在 变量 前 使 用 关键 字 static 声明 变量 。 格 式 如 下 : 

static $str; 

例 2.15 下 面 应 用 静态 变量 和 普通 变量 同时 输出 一 个 数据 ， 看 两 者 的 功能 有 什么 不 同 。 (实例 位 置 ; 
光盘 \TM\Instance\02\2.15) 


实例 代码 如 下 : 

<?php 

function zdy (X{ 
static $message = 0 ; // 初 始 化 静态 变量 
$message+=1; /静态 变 量 加 1 
echo $message." "; /| 输出 静态 变量 


全 
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function zdy1(){ 
Smessage = 0; /声明 函数 内 部 变量 (局 部 变量 ) 
$message += 1 : /局 部 变量 加 1 
echo $message"" // 输 出 局 部 变量 

} 

for ( $i=0 ; $i<10 ; $i++ ) zdy() ; /输出 1~10 

echo "<br>"; 

for ($i=0 ; $i<10 ; $i++ ) zdy1() ; 1/ 输 出 10 个 1 

?> 


运行 结果 如 图 2.10 所 示 。 

自 定义 函数 zdy0 是 输出 1 一 10 的 10 个 数字 ， 而 zdy10 函 
数 则 输出 的 是 10 个 1。 因 为 自 定义 函数 zdy0 含 有 静态 变量 ， 
而 函数 zdy10 是 一 个 普通 变量 。 初始化 都 为 0， 再 分 别 使 用 for 
循环 调用 两 个 函数 ， 结 果 是 静态 变量 的 函数 zdy0 在 被 调用 后 
保留 了 $message 中 的 值 , 静态 变量 的 初始 化 只 是 在 第 一 次 遇 到 
时 被 执行 ， 以 后 就 不 再 对 其 进行 初始 化 操作 了 ， 将 会 略 过 第 三 


图 2.10 比较 静态 变量 和 普通 变量 的 区 别 
行 代码 不 执行 ; 而 普通 变量 的 函数 zdy10 在 被 调用 后 , 其 变量 Smessage 失去 了 原来 的 值 , 重新 被 初始 化 为 0。 


2.4.3 可 变 变量 


可 变 变量 是 一 种 独特 的 变量 ， 变 量 的 名 称 并 不 是 预先 定义 好 的 ， 而 是 动态 地 设置 和 使 用 。 可 变 变量 一 
般 是 指使 用 一 个 变量 的 值 作为 另 一 个 变量 的 名 称 ， 所 以 可 变 变量 又 称 为 变量 的 变量 。 可 变 变 量 通过 在 一 个 
变量 名 称 前 使 用 两 个 “$” 符 号 实现 。 

例 2.16 下 面 应 用 可 变 变 量 实现 动态 改变 变量 的 名 称 。 首 先 定义 两 个 变量 $change_name 和 $php， 并 且 
输出 变量 Schange_name 的 值 ， 然 后 应 用 可 变 变 量 来 改变 变量 Schange_name 的 名 称 ， 最 后 输出 改变 名 称 后 的 
变量 值 。〈 实 例 位 置 ， 光盘 \TMNVInstance\02\2.16) 

程序 代码 如 下 : 


<?phl 

Shang neme = "Look"; /| 声明 变量 $change_name 
$Look = "美好 的 一 天 开始 了 ""; // 声 明 变 量 $php 

echo $change_name ; /| 输出 变量 $change_name 
echo $$change_name ; // 通 过 可 变 变量 输出 $php 的 值 
?> 


结果 为 : Look 美好 的 一 天 开始 了 ! 
2.4.4 ”预定 义 变量 

PHP 还 提供 了 很 多 非常 实用 的 预定 义 变量 ， 通 过 这 些 预 定义 变量 可 以 获取 到 用 户 会 话 、 用 户 操作 系统 
的 环境 和 本 地 操作 系统 的 环境 等 信息 。 常 用 的 预定 义 变量 如 表 2.9 所 示 。 


表 2.9 预定 义 变量 


变量 的 名 称 
$_SERVER['SERVER ADDR'] 


说 明 
当前 运行 脚本 所 在 服务 器 的 他 地 址 
当前 运行 脚本 所 在 服务 器 主机 的 名 称 。 如果 该 脚本 运行 在 一 个 虚拟 主机 上 ， 则 该 名 
称 是 由 那个 虚拟 主机 所 设置 的 值 决定 


$_SERVER['SERVER NAME'] 
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续 表 

变量 的 名 称 说 明 
访问 页 面 时 的 请 求 方法 , 如 GET、HEAD、POST、PUT。 如 果 请 求 的 方式 是 HEAD， 
则 PHP 脚本 将 在 送出 头 信息 后 终止 (这 意味 着 在 产生 任何 输出 后 , 不 再 有 输出 缓冲 ) 
$ SERVER[REMOTE ADDR'] 正在 浏览 当前 页 面 用 户 的 了 P 地 址 
$ SERVER[REMOTE HOST 正在 浏览 当前 页 面 用 户 的 主机 名 。 反 向 域名 解析 基于 该 用 户 的 REMOTE_ADDR 
$_SERVER['REMOTE PORT'] 用 户 连 接 到 服务 器 时 所 使 用 的 端口 
当前 执行 脚本 的 绝对 路 径 名 。 注意， 如 果 脚 本 在 CLI 中 被 执行 ， 作 为 相对 路 径 ， 如 
filephp 或 者 ./filephp，$_SERVER['SCRIPT FILENAME'] 将 包含 用 户 指定 的 相对 路 径 
服务 器 所 使 用 的 端口 ， 默 认为 80。 如 果 使 用 SSL 安全 连接 ， 则 这 个 值 为 用 户 设置 
的 HTTP 端口 
$ SERVER[REMOTE ADDR' 正在 浏览 当前 页 面 用 户 的 他 地 址 
$_SERVER[DOCUMENT ROOT'] | 当前 运行 脚本 所 在 的 文档 根 目录 。 在 服务 器 配置 文件 中 定义 

通过 HTTPCookie 传递 到 脚本 的 信息 。 这 些 cookie 多 数 是 由 执行 PHP 脚本 时 通过 


$_SERVER['REQUEST METHOD'] 


$_SERVER['SCRIPT FILENAME'] 


$_SERVER['SERVER PORT'] 


$_COOKIE 


setcookie0 函 数 设 置 的 
包含 与 所 有 会 话 变 量 有 关 的 信息 。$_SESSION 变量 主要 应 用 于 会 话 控制 和 页 面 之 
$_SESSION . 
间 值 的 传递 
包含 通过 POST 方法 传递 的 参数 的 相关 信息 。 主 要 用 于 获取 通过 POST 方法 提交 的 
$_POST 数据 
$ GET 包含 通过 GET 方法 传递 的 参数 的 相关 信息 。 主 要 用 于 获取 通过 GET 方法 提交 的 数据 
人 由 所 有 已 定义 的 全 局 变量 组 成 的 数组 。 变 量 名 就 是 该 数组 的 索引 。 它 可 以 被 称 为 所 
有 超级 变量 的 超级 集合 


2.4.5 ”变量 的 生存 周期 


变量 存在 的 时 间 称 为 生存 周期 ， 即 从 变量 被 声明 的 那 一 刻 起 ， 直 到 脚本 运行 结束 。 对 于 过 程 级 变量 ， 
其 存活 期 仅 是 该 过 程 运行 的 时 间 ， 该 过 程 结 束 后 ， 变 量 随 之 消失 。 在 执行 过 程 中 ， 局 部 变量 是 理想 的 临时 
存储 空间 。 在 不 同 过 程 中 可 以 使 用 同名 的 局 部 变量 ， 这 是 因为 每 个 局 部 变量 只 被 声明 它 的 过 程 识 别 。 


二 信人 大 


2.3 ”PHP 运 和 由 和 付 


锋 4 视频 讲解 : 光盘 \IM\Video\ 第 2 章 \PHP 运算 符 .exe 

运算 符 是 用 来 对 变量 、 常 量 或 数据 进行 计算 的 符号 ， 它 对 一 个 值 或 一 组 值 执行 一 个 指定 的 操作 。PHP 
的 运算 符 包括 算术 运算 符 、 字 符 串 运算 符 、 赋 值 运算 符 、 递 增 或 递减 运算 符 、 位 运算 符 、 逻 辑 运算 符 、 比 
较 运算 符 和 条 件 运算 符 。 下 面 进行 详细 讲解 。 


2.5.1 算术 运算 符 


算术 运算 符 用 来 连接 运算 表达 式 。 算 术 运算 符 包括 加 (+) 、 减 (-) 、 乘 (*) 、 除 (/) 、 取 模 (%) 
等 运算 符 。 常 用 的 算术 运算 符 如 表 2.10 所 示 。 


局 


算术 操作 符 
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表 2.10 常用 的 算术 运算 符 


举例 〈 定 义 $a=5，$b=3) 结 果 


Sat+$b 值 为 8 


$a-$b 值 为 2 


Sa* $b 值 为 15 


DE 


$a/$b 值 为 1.66 
值 为 2 


在 算术 运算 符 中 使 用 % 求 余 ， 如 果 被 除数 ($a ) 是 负数 的 话 ， 那 么 取得 的 结果 也 是 一 个 负 值 。 


例 2.17 在 本 例 中 分 别 应 用 算术 运算 符 对 表达 式 进行 计算 。〔 实 例 位 置 ， 光盘 \TM\Instance\02\2.17) 
实例 代码 如 下 : 
<?php 


?> 


$a=6; 

$b=-4; 

$c = 10; 

echo "$a = ".$a."<br>", 

echo \$b = ".$b."<br>"; 

echo "$c =".$c."<br>"; 

echo "$a + \$b = ".($a + $b)."<br>"; 
echo "\$a - \$b = ".($a - $b)."<br>";; 
echo “$a * \$b = ".($a * $b)."<br>"; 
echo "\$a / \$b = ".($a / $b)."<br>"; 
echo "\$a % \$c = ".($a % $c); 


运行 结果 如 图 2.11 所 示 。 


之 .2 


字符 串 运 算 符 只 有 一 个 ， 即 英文 的 句号 “.”。 它 将 两 个 
字符 串 连接 起 来 ， 形 成 一 个 新 的 字符 串 。 使 用 过 C 或 Java 的 
读者 要 注意 了 ， 这 里 的 “+” 号 ， 


字符 串 运 算 符 


能 做 字符 串 运 算 符 。 
例 2.18 下 面 应 用 一 个 实例 ， 来 看 一 下 (.) 和 “+” 两 者 


之 间 的 


/声明 变量 $a 

/声明 变量 $b 

/声明 变量 $c 

/| 输出 变量 $a 

// 输 出 变量 $b 

/| 输出 变量 $c 

// 计 算 变 量 $a 加 $b 的 值 
// 计 算 变 量 $a 减 $b 的 值 
// 计 算 $a 乘 以 $b 的 值 

// 计 算 $a 除 以 $b 的 值 
/计算 $a 和 $c 的 余数 ， 被 除数 为 15 


mb) 


只 做 赋值 运算 符 使 用 ， 而 不 ”一 Saezam 病 


图 2.11 算术 运算 符 的 应 用 


区 别 。 当 使 用 “.” 时 ， 变 量 Sm 和 $n 两 个 字符 串 组 成 一 个 新 的 字符 “3.5el1”， 当 使 用 “+” 时 ，PHP 


会 认为 这 是 一 次 运算 。 如 果 “+” 号 的 两 边 有 字符 类 型 ， 则 自动 转换 为 整 型 ， 如 果 是 字母 ， 则 输出 为 0， 如 
果 是 以 数字 开头 的 字符 串 , 则 会 截取 字 串 头 部 的 数字 , 再 进行 运算 。( 实 例 位 置 : 光盘 \TM\Instance\02\2.18) 
实例 代码 如 下 : 
<?php 
$n = "3.5e"; 
$m =11; 
$nm = $n.$m; 
echo $nm."\t"; 
$mn = $n + $m ; 
echo $mn; 


?> 


// 声 明 一 个 字符 串 变量 ， 以 数字 开头 
/声明 一 个 整 型 变量 
/使 用 “.” 运 算 符 将 两 个 变量 连接 


/使 用 “+” 运 算 符 将 两 个 变量 连接 
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结果 为 : 3.5ell 14.5。 
2.5.3 ”赋值 运算 符 
最 基本 的 赋值 运算 符 是 “=”， 用 于 对 变量 进行 赋值 ， 而 其 他 运算 符 可 以 和 赋值 运算 符 “=” 联 合 使 用 ， 


构成 组 合 赋值 运算 符 。 赋 值 运算 符 是 把 基本 赋值 运算 符 “=” 右 边 的 值 赋 给 左边 的 变量 或 者 常量 。 在 PHP 
中 的 赋值 运算 符 如 表 2.11 所 示 。 


表 2.11 常用 赋值 运算 符 


意 义 


赋值 将 右边 表达 式 的 值 赋 给 左边 的 变量 
加 b | $a=$a +b | 将 运算 符 左边 的 变量 加 上 右边 表达 式 的 值 赋 给 左边 的 变量 
减 $a 一 b | $a=$a-b ”| 将 运算 符 左 边 的 变量 减 去 右边 表达 式 的 值 赋 给 左边 的 变量 
es | 将 运算 符 左边 的 变量 乘 以 右边 表达 式 的 值 赋 给 左边 的 变量 
| 所 | $a=b | $a=$a/b | 将 运算 符 左边 的 变量 除 以 右边 表达 式 的 值 赋 给 左边 的 变量 
一 一 $a=$a.b | 将 右边 的 字符 加 到 左边 
| %= | $a%=b | $a=$a %b | 将 运算 符 左边 的 变量 用 右边 表达 式 的 值 求 模 ， 并 将 结果 赋 给 左边 的 变量 


例 2.19 应 用 赋值 运算 符 给 指定 的 变量 赋值 ， 并 计算 各 表达 式 的 值 。( 实 例 位 置 ; 光盘 \TMNInstance\02\2.19) 


代码 如 下 : 
<?php 
$a=8; // 声 明 变量 $a 
$b =4; // 声 明 变量 $b 
echo "$a = ".$a."<br>"; // 输 出 变量 $a 
echo "$b = ".$b."<br>"; // 输 出 变量 $b 
echo "$a += \$b = ".($a += $b)."<br>"; // 计 算 变量 $a 加 $b 的 值 
echo "\$a -= \$b = ".($a -= $b)."<br>"; // 计 算 变 量 $a 减 $b 的 值 
echo "$a *= \$b = ".($a *= $b)."<br>"; /计算 $a 乘 以 $b 的 值 
echo "$a /= \$b = ".($a /= $b)."<br>"; /计算 $a 除 以 $b 的 值 
echo "$a %= \$b = ".($a %= $b); // 计 算 $a 和 $b 的 余数 
?> 


运行 结果 如 图 2.12 所 示 。 


2.5.4 ”递增 或 递减 运算 符 


BEE Wndowr raemer rplorer 


Ge eam - 


算术 运算 符 适合 在 有 两 个 或 者 两 个 以 上 不 同 操作 数 的 场 
合 使 用 ， 但 是 ， 当 只 有 一 个 操作 数 时 ， 使 用 算术 运算 符 是 没有 
必要 的 。 这 时 ， 就 可 以 使 用 “++” 或 者 “一 ”运算 符 了 ， 即 递 
增 或 递减 运算 符 。 图 2.12 赋值 运算 符 的 应 用 

递增 或 递减 运算 符 有 两 种 使 用 方法 ， 一 种 是 先 将 变量 增加 或 者 减少 1 后 再 将 值 赋 给 原 变 量 ， 称 为 前 置 
递增 或 递减 运算 符 ( 也 称 前 置 自 增 自 减 运 算 符 ) ;， 另 一 种 是 将 运算 符 放 在 变量 后 面 ， 即 先 返回 变量 的 当前 
值 ， 然 后 变量 的 当前 值 增加 或 者 减少 1， 称 为 后 置 递增 或 递减 运算 符 〈 后 置 自 增 自 减 运算 符 ) 。 

1. 前 置 递增 运算 符 


前 置 递增 运算 符 是 指使 用 前 置 “++” 运 算 符 将 变量 加 1 再 将 值 赋 给 原 变量 。 例 如 : 
<?php 


局 


ET ET 而- S100% ~ 
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$a=5; 
echo ++$a; 
?> 


结果 为 : 6。 
该 程序 的 运行 结果 为 6， 也 就 是 首先 把 变量 $a 加 1， 再 将 结果 赋 给 原 变量 ， 所 以 运行 本 程序 后 ， 原 变量 


$a 的 值 已 经 变 成 6。 


2. 前 置 递 减 运算 符 


前 置 递减 运算 符 是 指使 用 前 置 “ 一 ”运算 符 将 变量 减 1 再 将 值 赋 给 原 变量 。 例 如 : 
<?php 
$a=5; 
echo —$a; 
?> 
结果 为 : 4。 
该 程序 的 运行 结果 为 4， 也 就 是 首先 把 变量 $a 减 1， 再 将 结果 赋 给 原 变量 ， 所 以 运行 本 程序 后 ， 原 变量 


$a 的 值 已 经 变 成 4。 


3. 后 置 递 增 运算 符 


后 置 递增 运算 符 是 指 先 返回 变量 的 当前 值 ， 然后 再 使 用 后 置 “++” 运 算 符 将 变量 的 当前 值 再 加 1。 例如 : 
<?php 
$a=5; 
echo S$a+tt+; 
Ee 
结果 为 ，5。 
该 程序 的 运行 结果 为 5， 即 先 将 变量 的 原 值 输出 ， 然 后 再 加 1， 虽 然 显 示 输出 的 是 5， 但 实际 上 变量 $a 


的 值 已 经 是 6 了 。 


4. 后 置 递减 运算 符 
后 置 递减 运算 符 是 指 先 返 回 变量 的 当前 值 ， 然 后 使 用 后 置 “--” 运 算 符 将 变量 的 当前 值 再 减 1。 例 如 : 


结果 为 : 5。 
该 程序 的 运行 结果 为 5， 即 先 将 变量 的 原 值 输出 ， 然 后 再 减 1， 虽 然 显示 输出 的 是 5， 但 实际 上 变量 $a 


的 值 已 经 是 4 了。 


2.5.5 ”位 运算 符 


位 运算 符 是 指 对 二 进 制 位 从 低位 到 高 位 对 齐 后 进行 运算 。 在 PHP 中 的 位 运算 符 如 表 2.12 所 示 。 


表 2.12 位 运算 符 

运 算 符 举例 
到 $m & $n 
| Sm | $n 
入 Sm 人 $n 
sd Sm ~ $n 
<< $m << $n 
>> $m >> $n 
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例 2.20 ”应 用 位 运算 符 对 变量 中 的 值 进行 位 运算 操作 。〈 实 例 位 置 : 光盘 \TMNInstance\02\2.20) 
实例 代码 如 下 : 


Sn =8 ; 

$mn = $m & $n ; /位 与 
echo $mn ." \t"; 

Smn = $m | $n; /位 或 
echo $mn ." \t"; 

$mn = $m $n ; // 位 异 或 
echo $mn ." \t"; 

$mn = ~$m; /位 取 反 
echo $mn ; 

rt 


结果 为 : 0 15 15 -8。 
2.5.6 ”逻辑 运算 符 


逻辑 运算 符 用 来 组 合 逻 辑 运算 的 结果 ， 是 程序 设计 中 一 组 非常 重要 的 运算 符 。PHP 的 逻辑 运算 符 
如 表 2.13 所 示 。 
表 2.13 PHP 的 逻辑 运算 符 


: 结果 为 真 
当 Sm 和 Sn 都 为 真 时 
当 Sm 为 真 或 者 Sn 为 真 时 
当 Sm、Sn 一 真一 假 时 
当 Sm 为 假 时 


在 逻辑 运算 符 中 ， 逻 辑 与 和 逻辑 或 这 两 个 运算 符 有 4 种 运算 符号 (&&、and、|| 和 or) ， 其 中 属于 同一 
个 逻辑 结构 的 两 个 运算 符号 (如 && 和 and) 之 间 却 有 着 不 同 的 优先 级 。 
例 2.21 下 面 分 别 应 用 逻辑 或 中 的 运算 符号 “||” 和 “or” 进 行 相同 的 判断 ， 但 是 因为 同一 逻辑 结构 的 


两 个 运算 符 〈“||” 和 “or”) 的 优先 级 不 同 ， 输 出 的 结果 也 不 同 。〈 实 例 位 置 : 光盘 \TMNInstance\02\2.21) 
实例 代码 如 下 : 
<?php 
$i = true; /声明 一 个 布尔 变量 $i， 赋 值 为 真 
$j = true; /再 声明 一 个 布尔 变量 $j， 赋 值 也 为 真 
$z = false; // 最 后 再 声明 一 个 初 值 为 假 的 布尔 变量 $z 
if($i or $j and $z) // 这 是 用 or 做 判断 
echo "true"; // 如 果 if 表 达 式 为 真 ， 输 出 true 
else 
echo "false"; // 否 则 输出 false 
echo "<br>"; 
if($i || $j and $z) // 这 是 用 || 做 判断 
echo "true"; // 如 果 表 达 式 为 真 ， 输 出 true 
else 
echo "false"; // 如 果 表达 式 为 假 ， 输 出 false 
?> 
结果 为 : trmue 
false。 


@ 


第 2 章 PHP 语言 基础 


ea 


在 实际 应 用 中 要 多 注意 这 样 的 细节 。 


2.5.7 ”比较 运算 符 


比较 运算 符 就 是 对 变量 或 表达 式 的 结果 进行 大 小 、 真 假 等 比较 ， 如 果 比 较 结果 为 真 ， 则 返 


为 假 ， 则 返回 false。PHP 中 的 比较 运算 符 如 表 2.14 所 示 。 


表 2.14 PHP 的 比较 运算 符 


回 true; 如 果 


运 算 符 举例 定义 Sm=5，S$n=2) 结果 
< Sm<Sn false 
Sm>S$n true 
<= S$m<=$n false 
>= Sm>=$n tme 
三 三 Sm= =$n false 
丘 Sm!=S$n true 
=== Sm=== $n false 
= Sm!=—$n tme 
比较 运算 “ “Sm=== $n”， 说 明 $m 和 Sn 不 只 是 数值 上 相等 ， 而 且 两 者 
的 类 型 也 一 样 ”和 “===” 的 含义 差不多 ，“$m != = $n” 是 指 Sm 和 $n 或 者 数值 不 相等 ， 或 者 类 


型 不 相等 。 
例 2.22 下 面 应 用 比较 运算 符 对 变量 中 的 值 进行 比较 ， 设 置 变量 $value ="100"， 变 量 的 类 型 为 字符 串 
型 ,将 变量 $value 与 数字 100 进行 比较 , 会 发 现 比较 的 结果 非常 有 趣 。 其 中 使 用 的 var_dump0 函 数 是 系统 函 


数 ， 


作用 是 输出 变量 的 相关 信息 。 


实例 代码 如 下 : 

<?php 

S$value="100"; 

echo "\$value = \"$value\"™"; 
echo "<p>\$value==100: "; 
var_dump($value==100); 
echo "<p>\$value==true: "; 
var_dump($value==true); 
echo "<p>\$valuel=null: "; 
var_dump($valuel=null); 
echo "<p>\$value==false: "; 


(实例 位 置 ， 光盘 \TM\Instance\02\2.22) 


// 声 明 一 个 字符 串 变量 $value 


/| 结果 为 : bool(true) 
/| 结果 为 : bool(true) 
/| 结果 为 : bool(true) 
/| 结果 为 : bool(false) 
/| 结果 为 : bool(false) 
/| 结果 为 : bool(false) 


/| 结果 为 : bool(true) 
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运行 结果 如 图 2.13 所 示 。 
2.5.8 条 件 运算 符 


条 件 运 算 符 用 于 根据 一 个 表达 式 在 另 两 个 表达 式 中 选择 
一 个 , 而 不 是 用 来 在 两 个 语句 或 者 程序 中 选择 。 条 件 运算 符 也 
称 三 元 运算 符 ， 条 件 运算 符 的 使 用 最 好 放 在 括号 里 。 

例 2.23 下面 应 用 条 件 运 算 符 实现 一 个 简单 的 判断 功能 ， 
如 果 变 量 $age 的 值 大 于 等 于 9， 则 输出 “小 红 是 好 人 ”， 否 则 
输出 “小 红 是 坏人 ”。 (实例 位 置 : 光盘 \TMNInstance\02\2.23) 图 2.13 ”比较 运算 符 的 应 用 

实例 代码 如 下 : 

<?php 

$age=12; 

echo ($age>=9)? 小 红 是 好 人 : 小 红 是 坏人 ; 

?> 


结果 为 ， 小 红 是 好 人 。 
2.5.9 ”运算 符 的 优先 顺序 和 结合 规则 


所 谓 运 算 符 的 优先 级 ， 是 指 在 表达 式 中 哪 一 个 运算 符 先 计算 ， 哪 一 个 后 计算 ， 与 数学 的 四 则 运算 遵循 
的 “ 先 乘除 ， 后 加 减 ” 是 一 个 道理 。 

PHP 的 运算 符 在 运算 中 遵循 的 规则 是 : 优先 级 高 的 操作 先 执行 ， 优 先 级 低 的 操作 后 执行 ， 同 一 优先 级 
的 操作 按照 从 左 到 右 的 顺序 进行 。 也 可 以 像 四 则 运算 那样 使 用 小 括号 ， 括 号 内 的 运算 最 先进 行 。PHP 运算 
符 优 先 级 的 一 览 表 如 表 2.15 所 示 。 


表 2.15 运算 符 的 优先 级 


优先 级 别 运 算 符 
1 or、and、xor 
2 =、 +=、 一 、*=、 人 =、=、%= 
3 | && 
4 a 
&、. 
6 ++、 一 (递增 或 递减 运算 符 ) 
/~ *、% 
8 <<、>> 
9 二 
10 + -〈 正 、 负 号 运算 符 ) 、!、 一 
11 Ey 
i AS 
13 ?: 
14 一 
15 => 
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2.6 PHP 函数 


区 il 视频 讲解 :光盘 \TM\Video\ 第 2 章 \PHP 函数 .exe 
在 开发 过 程 中 ， 经 常 要 重复 某 种 操作 或 处 理 ， 如 数据 查询 、 字 符 操作 等 ， 如 果 每 个 模块 的 操作 都 要 重 


新 输入 一 次 代码 ， 不 仅 令 程序 员 头 痛 不 已 ， 而 且 对 于 代码 的 后 期 维护 及 运行 效果 也 有 着 较 大 的 影响 ， 使 用 
PHP 函数 即 可 让 这 些 问题 迎刃而解 。 下 面 将 介绍 这 些 知识 。 


2.6.1 ”定义 和 调用 函数 


函数 ， 就 是 将 一 些 重复 使 用 到 的 功能 写 在 一 个 独立 的 代码 块 中 ， 在 需要 时 单独 调用 。 创 建 函数 的 基本 
语法 格式 为 : 

function fun_name($str1,$stgr2**$strmm}{ 

fun_body; 

其 中 ， 
function: 为 声明 自 定义 函数 时 必须 使 用 到 的 关键 字 。 
fun_name: 为 自 定义 函数 的 名 称 。 
$str1…$stm:， 为 函数 的 参数 。 
fun_body: 为 自 定义 函数 的 主体 ， 是 功能 实现 部 分 。 

当 函 数 被 定义 好 后 ， 所 要 做 的 就 是 调用 这 个 函数 。 调 用 函数 地 操作 十 分 简单 ， 只 需要 引用 函数 名 并 赋 
予 正确 的 参数 即 可 完成 函数 的 调用 。 

例 2.24 在 本 例 中 定义 了 一 个 函数 example0, 计算 传 入 参数 的 平方 , 然后 连同 表达 式 和 结果 全 部 输出 。 
(实例 位 置 ， 光 盘 \TM\Instance\02\2.24) 


国 办 办 办 


实例 代码 如 下 : 
<?php 
”声明 自 定义 函数 */ 
function example($num){ 
return "$num * $num = ".$num * $num; // 返 回 计算 后 的 结果 
} 
echo example(5); // 调 用 函数 


re 
结果 为 : 5 * 5 =25。 


2.6.2 ”在 函数 间 传 递 参数 


在 调用 函数 时 ， 需 要 向 函数 传递 参数 ， 被 传 入 的 参数 称 为 实 参 ， 而 函数 定义 的 参数 为 形 参 。 参 数 传递 
的 方式 有 按 值 传递 、 按 引用 传递 和 默认 参数 3 种 。 


1. 按 值 传递 方式 


将 实 参 的 值 复制 到 对 应 的 形 参 中 ， 在 函数 内 部 的 操作 针对 形 参 进行 ， 操 作 的 结果 不 会 影响 到 实 参 ， 即 
函数 返回 后 ， 实 参 的 值 不 会 改变 。 


例 2.25 本 例 首先 定义 一 个 函数 example0， 功 能 是 将 传 入 的 参数 值 做 一 些 运算 后 再 输出 。 接 着 在 函数 


® 
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外 部 定义 一 个 变量 gm， 也 就 是 要 传 进来 的 参数 。 最 后 调用 函数 example($m)， 输 出 函数 的 返回 值 gm 和 变量 
$m 的 值 。 (实例 位 置 : 光盘 \TM\Instance\02\2.25) 


实例 代码 如 下 : 
<?php 
function example( $m X{ /定义 一 个 函数 
Sm= $m*4+8; 
echo "在 函数 内 : \$m = ".$m; /输出 形 参 的 值 
8 
$m=1; 
example( $m ) ; /传递 值 ， 将 $m 的 值 传 递 给 形 参 $m 
echo "<p> 在 函数 外 \$m = $m <p>"; // 实 参 的 值 没有 发 生变 化 ， 输 出 m=1 
3 


运行 结果 如 图 2.14 所 示 。 

2. 按 引用 传递 方式 

按 引 用 传递 就 是 将 实 参 的 内 存 地址 传递 到 形 参 中 。 这 时 ， 在 函数 内 部 的 所 有 操作 都 会 影响 到 实 参 的 值 ， 
返回 后 ， 实 参 的 值 会 发 生变 化 。 引 用 传递 方式 就 是 传 值 时 在 原 基础 上 加 & 号 即 可 。 

例 2.26 仍然 使 用 例 2.25 中 的 代码 ,唯一 不 同 的 地 方 就 是 多 了 一 个 & 号 。( 实 例 位 置 : 光盘 \TM\Instance\ 
02\2.26) 


实例 代码 如 下 : 

<?php 

function example( &$m X{ // 定 义 一 个 函数 ， 同 时 传递 参数 $m 的 变量 
Sm= $m*4+8; 

echo "在 函数 内 : \$m = ".$m; /输出 形 参 的 值 

上 

$m=1; 

example( $m ) ; /传递 值 : 将 $m 的 值 传递 给 形 参 $m 

echo "<p> 在 函数 外 : \$m = $m <p>"; // 实 参 的 值 发 生变 化 ， 输 出 m=15 

?> 


运行 结果 如 图 2.15 所 示 。 
大 元 本 是 六 四 -windowe intemer Explorer 
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全 本 地 Intranec | 出 P 杰 式 :各 用 L 
图 2.14 接 值 传递 方式 图 2.15 按 引 用 传递 方式 


3. 默认 参数 〈 可 选 参数 ) 

还 有 一 种 设置 参数 的 方式 ， 即 可 选 参数 。 可 以 指定 某 个 参数 为 可 选 参数 ， 将 可 选 参数 放 在 参数 列表 末 
并 且 指 定 其 默认 值 为 空 。 

例 2.27 本 例 使 用 可 选 参数 实现 一 个 简单 的 价格 计算 功能 , 设置 自 定 义 函 数 values 的 参数 Stax 为 可 选 
参数 ， 其 默认 值 为 空 。 第 一 次 调用 该 函数 ， 并 且 给 参数 $tax 赋值 036， 输 出 价格 ; 第 二 次 调用 该 函数 ， 不 给 
参数 $tax 赋值 ， 输 出 价格 。〔 实 例 位 置 ， 光 盘 \TM\Instance\02\2.27) 


尾 


实例 代码 如 下 : 
<?php 
function values($price,$tax="")f /定义 一 个 函数 ， 其 中 的 一 个 参数 初始 值 为 空 
S$price=$price+($price*S$tax); /声明 一 个 变量 $price， 等 于 两 个 参数 的 运算 结果 


局 
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echo "价格 :$price<br> /输出 价格 
} 
values(100,0.36); /为 可 选 参数 赋值 0.36 
values(100); /没有 给 可 选 参数 赋值 


?> 


结果 为: 价格 :136 ”价格 :100。 
《人 注意 当 使 用 默认 参数 时 ， 默 认 参 数 必 须 放 在 非 默认 参数 的 右 侧 ， 和 否则 函数 可 能 出 错 。 
by B 

说 明 从 PHp5 开始 ， 默 认 值 也 可 以 通过 引用 传递 。 


2.6.3 ”从 函数 中 返回 值 


在 2.6.1 节 中 介绍 了 如 何 定义 和 调用 一 个 函数 ， 并 且 讲 解 了 如 何在 函数 间 传递 值 ， 本 节 将 讲解 函数 的 返 
回 值 。 通 常 ， 函 数 将 返回 值 传递 给 调用 者 的 方式 是 使 用 关键 字 return。 

retur 将 函数 的 值 返 回 给 函数 的 调用 者 ， 即 将 程序 控制 权 返 回 到 调用 者 的 作用 域 。 如 果 在 全 局 作用 域内 
使 用 retum 关键 字 ， 那 么 将 终止 脚本 的 执行 。 

例 2.28 本 实例 使 用 return0 函 数 返 回 一 个 操作 数 。 先 定义 函数 values0, 函数 的 作用 是 输入 物品 的 单价 、 
重量 ， 然 后 计算 总 金额 ， 最 后 输出 商品 的 价格 。〔 实 例 位 置 ， 光盘 \TM\Instance\02\2.28) 


实例 代码 如 下 : 
<?php 
function values($price, $tax=0.25){ /定义 一 个 函数 ， 函 数 中 的 一 个 参数 有 默认 值 
S$price=$price+($price*$tax); /计算 物品 金额 
return $price; // 返 回 金 额 
} 
echo values(100); // 调 用 函数 
?> 
结果 为 :125。 


retum 语句 只 能 返回 一 个 参数 ， 也 只 能 返回 一 个 值 ， 不 能 一 次 返回 多 个 。 如 果 要 返回 多 个 结果 ， 就 要 在 
函数 中 定义 一 个 数组 ， 将 返回 值 存储 在 数组 中 返回 。 


2.6.4 变量 函数 


PHP 支持 变量 函数 。 下 面 通过 一 个 实例 来 介绍 变量 函数 的 具体 应 用 。 

例 2.29 ”本 例 首先 定义 了 3 个 函数 ， 接 着 声明 一 个 变量 ， 通 过 变量 来 访问 不 同 的 函数 。 (实例 位 置 
光盘 \TM\Instance\02\2.29) 

实例 代码 如 下 : 


<?php 

function come() { /定义 come() 函 数 
echo "来 了 <p>"; 

function go($name = "jack") { /定义 go() 函 数 


echo " $name. ' 走 了 <p>"; 
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} 
function back($string) /定义 back() 函 数 


echo "又 回来 了 ，S$string<p>"; 
} 


S$func = "come"; /声明 一 个 变量 ， 将 变量 赋值 为 “come” 
Sfunc(); /使 用 变量 函数 来 调用 函数 come() 
S$func = "go" /重新 给 变量 赋值 

$func("Tom"); /使 用 变量 函数 来 调用 函数 go() 

$func = "back"; /重新 给 变量 赋值 

S$func("Lily /使 用 变量 函数 来 调用 函数 back() 

?> 

运行 结果 如 图 2.16 所 示 。 


二 window Internet Excplorer 
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可 以 看 到 ,函数 的 调用 是 通过 改变 变量 名 来 实现 的 , 通过 
在 变量 名 后 面 加 上 一 对 小 括号 , PHP 将 自动 寻找 与 变量 名 相同 
的 函数 , 并 且 执 行 它 。 如 果 找 不 到 对 应 的 函数 , 系统 将 会 报错 。 
这 个 技术 可 以 用 于 实现 回调 函数 和 函数 表 等 。 


2.6.5 ”对 函数 的 引用 图 2.16 变量 函数 


在 2.6.2 节 的 参数 传递 中 有 按 引 用 传递 的 方式 ， 可 以 修改 实 参 的 内 容 。 引 用 不 仅 用 于 普通 变量 、 函 数 参 
数 ， 也 可 作用 于 函数 本 身 。 对 函数 的 引用 ， 就 是 对 函数 返回 结果 的 引用 。 

例 2.30 在 本 例 中 ,首先 定义 一 个 函数 ， 这 里 需 在 函数 名 前 加 “&” 号 ,接着 ,变量 $str 将 引用 该 函数 ， 
最 后 输出 该 变量 $str， 实 际 上 就 是 $tmp 的 值 。 (实例 位 置 : 光盘 \TM\Instance\02\2.30) 


实例 代码 如 下 : 

<?php 

function &example($tmp=0){ // 定 义 一 个 函数 ， 加 “&” 号 
return $tmp; // 返 回 参数 Stmp 

} 

$str = &example(" 明 日 科技 "); // 声 明 一 个 函数 的 引用 $str 

echo $str."<p>"; 1/ 输 出 $str 

?> 


结果 为 : 明日 科技 。 
和。 注意 和 参数 传递 不 同 ， 这 里 必须 在 两 个 地 方 使 用 “&” 号 ， 用 来 说 明 返回 的 是 一 个 引用 。 
2.6.6 取消 引用 

当 不 再 需要 引用 时 ， 可 以 取消 引用 。 取 消 引用 使 用 unset0 函 数 ， 它 只 是 断 开 了 变量 名 和 变量 内 容 之 间 
的 绑 定 ， 而 不 是 销毁 变量 内 容 。 


例 2.31 本 例 首 先 声 明 一 个 变量 和 变量 的 引用 ， 输 出 引用 后 取消 引用 ， 再 次 调用 引用 和 原 变量 。 可 以 
看 到 ， 取 消 引 用 后 对 原 变量 没有 任何 影响 。 (实例 位 置 ， 光盘 \TMNInstance\02\2.31) 


实例 代码 如 下 : 
<?php 
$num = 789; // 声 明 一 个 整 型 变量 
$math = &$num; /声明 一 个 对 变量 $num 的 引用 $math 


局 
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echo “$math is: ".$math."<br>"; /输出 引用 $math 
unset($math); /取消 引用 $math 
echo “$math is: ".$math."<br>"; // 再 次 输出 引用 
echo "$num is: "Snum; // 输 出 原 变量 

?> 

运行 结果 如 图 2.17 所 示 。 


图 2.17 取消 引用 


27 输出 语句 


顽 4 视频 讲解 : 光盘 \TM\Video\ 第 2 章 \ 输 出 语句 .exe 


程序 设计 中 经 常 需要 将 字符 串 或 者 字符 串 变量 输出 到 浏览 器 。 本 节 将 介绍 几 种 输出 语句 的 方法 。 由 于 
PHP 中 数据 类 型 是 可 以 自动 转换 的 ， 所 以 这 些 方法 也 可 用 于 输出 其 他 类 型 的 数据 。 


2.7.1 应 用 print 语句 输出 字符 


print 输出 语句 的 作用 是 将 字符 串 输出 到 浏览 器 或 打印 机 等 输出 设备 ， 通 常 输出 到 浏览 器 。print 语句 只 
可 以 同时 输出 一 个 字符 串 ， 需 要 圆 括号 。echo 语句 可 以 同时 输出 多 个 字符 串 ， 并 不 需要 圆 括号 。print 语句 
和 echo 语句 的 功能 是 相同 的 ， 但 print 语句 有 返回 值 ， 而 echo 语句 没有 返回 值 ， 所 以 echo 语句 相对 可 能 快 
些 ， 但 这 种 速度 差异 并 不 明显 。 

语法 : 

int print (string arg) 

该 语句 执行 成 功 则 返回 1， 失 败 则 返回 0。 


4 


全 5 注意 使 用 print 语 句 时， 括号 可 以 省 略 。 


例 2.32 应 用 print 语句 输出 字符 串 “PHP 编程 词典 , 让 您 编程 无 忧 ”。( 实 例 位 置 : 光盘 \TM\Instance\02\2.32) 

代码 如 下 : 

<?php 

print ("PHP 编程 词典 ， 让 您 编程 无 忧 "); 

?> 

运行 结果 如 图 2.18 所 示 。 

例 2.33 ”上面 讲解 了 应 用 print 语句 输出 字符 串 的 方法 , 接 下 来 应 用 print 语句 输出 变量 中 的 字符 串 .( 实 
例 位 置 ， 光盘 \TM\Instance\02\2.33) 

代码 如 下 : 

<?php 


$str=" 了 明日 图 书 辉煌 打造 "; 
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print $str; 
?> 


运行 结果 如 图 2.19 所 示 。 


图 2.18 应 用 print 语句 输出 字符 串 图 2.19 应 用 print 语句 输出 变量 中 的 字符 串 


例 2.34 应 用 print 语句 输出 新 型 字符 串 中 的 值 。 (实例 位 置 ， 光盘 \TM\Instance\02\2.34) 
代码 如 下 : 

<?php 

print <<<END 

吉林 省 明日 科技 有 限 公司 

END; 

i 


运行 结果 如 图 2.20 所 示 。 


a 由 于 这 是 一 个 语言 结构 ， 而 非 函数 ， 因 此 它 
无 法 被 变量 函数 调用 。 


2.7.2 ”应 用 echo 语句 输出 字符 


echo 语句 用 于 在 脚本 当前 位 置 输出 字符 串 ， 该 语句 在 前 面 章节 所 举 的 实例 中 多 次 使 用 过 ， 是 程序 设计 
中 最 常 使 用 的 一 种 输出 方法 。echo 使 用 方式 非常 简单 ， 直 接 将 要 输出 的 字符 串 或 者 变量 跟 在 echo 后 面 ， 字 
符 串 或 变量 和 echo 之 间 以 一 个 空格 分 隔 。 

语法 : 

echo "string arg1, string [arg2]..." 

echo 语句 会 将 传 入 的 字符 串 参数 (arg1) 进行 输出 。 由 于 它 本 身 并 不 是 一 个 真正 的 函数 , 因此 没有 返回 值 。 

使 用 echo 语句 执行 的 结果 是 ， 将 传递 给 其 自身 的 字符 串 输出 到 浏览 器 。 下 面 应 用 echo 语句 输出 字符 串 
“明日 科技 图 书 网 ，www.mingribook.com” 到 浏览 器 。 代 码 如 下 : 

<3| 

oS 了 有 日 科技 图 书 网 : www.mingribook.com" ; 

?> 

结果 为 ， 明 日 科技 图 书 网 www.mingribook.com。 

接 下 来 应 用 echo 语句 输出 变量 ， 代 码 如 下 。 

<?phi 

S$sting = "明日 编程 词典 网 ， www.mrbccd.com"; 

echo $string; 

?> 

结果 为 ， 明 日 编程 词典 网 : www.mrbccd.com。 


@ 


图 2.20 应 用 print 语句 输出 新 型 字符 串 中 的 值 
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echo 语句 后 面 可 以 连续 输出 多 个 字符 串 ， 它 们 之 间 以 逗号 分 隔 。 例 如 : 

<?php 

$mr=" 阴 日 科技 "; 

str=" 是 一 家 以 计算 机 软件 技术 为 核心 的 高 科技 型 企业 "; 

echo $mr,$str; 

这 

结果 为 : 明日 科技 是 一 家 以 计算 机 软件 技术 为 核心 的 高 科技 型 企业 。 

连续 输出 多 个 字符 串 ， 可 以 使 用 字符 串 运算 符 “.” 将 多 个 字符 串 连 接 成 一 个 新 型 字符 串 输出 。 例 如 : 

<?php 

S$str1=" 一 个 人 的 快乐 "; 

$str2=" 不 是 因为 他 拥有 的 多 "; 

$str3=" 而 是 因为 他 计较 的 少 。"; 

echo $str1.",".$str2.",".$str3."。" 

?> 

结果 为 : 一 个 人 的 快乐 ， 不 是 因为 他 拥有 的 多 ， 而 是 因为 他 计较 的 少 。 

echo 语句 后 紧 跟 的 参数 应 该 是 字符 串 类 型 ， 如 果 不 是 ，PHP 将 自动 将 其 进行 转换 ， 所 以 echo 语句 同样 
可 以 输出 其 他 类 型 的 数据 。 例 如 : 

<?php 

S$name=" 木 讷 "; 

$age="25"; 

$payfor="2400.00"; 

echo $name." 年 龄 : ".$age." 工资 :".$payfor; 

?> 

结果 为 : 木讷 年 龄 : 25 工资 : 2400.00。 

例 2.35 应 用 echo 语句 直接 输出 phpinfo0 函 数 调用 的 内 容 。 (实例 位 置 : 光盘 \TMNInstance\02\2.3S) 

代码 如 下 : 

<?php 

echo phpinfo(); 

?> 


结果 如 图 2.21 所 示 。 
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图 2.21 应 用 echo 语句 直接 输出 函数 中 的 内 容 


2.7.3 应 用 printf 语句 格式 化 输出 字符 


printf 语句 将 字符 串 按照 某 种 格式 进行 输出 。 
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语法 : 
int printf ( string format [, mixed args [, mixed ...]] ) 
printf 语句 按照 参数 format 指定 的 内 容 格式 对 字符 串 进行 格式 化 , 具体 参数 format 的 转换 格式 是 以 百 分 


比 符号 〈%) 开始 到 转换 字符 为 止 。 参 数 format 的 格式 转换 类 型 如 表 2.16 所 示 。 


表 2.16 参数 format 的 格式 转换 类 型 

说 明 
整数 转换 成 二 进 制 
整数 转换 成 对 应 的 ASCI 字 符 
整数 转换 成 十 进 制 
将 整数 转换 成 以 科学 计数 法 显示 
被 精确 的 数字 转换 成 浮 点 数 
整数 转换 成 八进制 
整数 转换 成 字符 串 
整数 转换 成 无 符号 整数 
整数 转换 成 小 写 十 六 进 制 
整数 转换 成 大 写 十 六 进 制 


例 2.36 应 用 printf 语句 将 整数 转换 成 不 同 的 类 型 ， 并 以 格式 化 的 方式 输出 字符 串 。《〈 实 例 位 置 ， 光盘 \ 
TM\Instance\02\2.36) 


“ls lv lo lmlo lalo Io 


代码 如 下 : 

<?php 

$string = 15; 

printf("%c",$string); // 将 整数 转换 成 ASCII 字符 
$string = "了 明日 科技 "; 

printf($string); // 直 接 输出 字符 

$string = 1234.00; 

printf("%0.2f",$string); // 将 整数 转换 成 小 数 点 后 有 两 位 小 数 的 浮 点 数 
printf("%s", $string); // 将 整数 转换 成 字符 串 
printf("%X",$string); // 将 整数 转换 成 大 写 十 六 进 制 
?> 


运行 结果 为 : 交 明 日 科技 1234.0012344D2。 
2.7.4 应 用 sprintf 语句 格式 化 输出 字符 


格式 化 字符 串 用 于 按 一 定 的 格式 输出 含有 许多 变量 的 文本 ， 是 最 常用 的 一 种 操作 ， 可 以 使 用 PHP 的 
sprintf 语 句 来 完成 格式 化 字符 串 的 功能 。 

语法 : 

string sprintf(string format, mixed args...) 

参数 format 为 转换 后 的 格式 ， 各 个 变量 都 以 “%” 后 的 字符 规定 其 格式 ， 格 式 转换 类 型 如 表 2.16 所 示 。 

例 2.37 应 用 sprintf 语句 格式 化 字符 串 。〈 实 例 位 置 ， 光盘 \TM\Instance\02\2.37) 

代码 如 下 : 

<?php 

$name= "木讷 "; 

S$pay= 2400; 

$expend= 154.5; 

S$balance= $pay-$expend; 

echo sprintf("%s: 您 的 本 月 工资 为 ¥Y %0.1f 元 ",$Sname,$balance) 

?> 


局 
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运行 结果 为 : 木讷 :您 的 本 月 工资 为 竺 2245.5 元 。 
2.8 引用 文件 


名 4 视频 讲解 : 光盘 \IM\Video\ 第 2 章 \ 引 用 文件 .exe 

引用 文件 是 指 将 另 一 个 源 文件 的 全 部 内 容 包含 到 当前 源 文件 中 进行 使 用 。 引 用 外 部 文件 可 以 减少 代码 
的 重用 性 , 是 PHP 编程 的 重要 技巧 。 PHP 提供 了 include 语句 require 语句 、include_once 语句 和 require_once 
语句 用 于 实现 引用 文件 。 这 4 种 语句 在 使 用 上 有 一 定 的 区 别 。 下 面 分 别 进行 详细 讲解 。 


2.8.1 应 用 include 语句 引用 文件 


使 用 include 语句 引用 外 部 文件 时 , 只 有 代码 执行 到 include 语句 时 才 将 外 部 文件 引用 进来 并 读 取 文件 的 
内 容 ， 当 所 引用 的 外 部 文件 发 生 错误 时 ， 系 统 只 给 出 一 个 警告 ， 而 整个 PHP 文件 则 继续 向 下 执行 。 下 面 介 
绍 include 语句 的 使 用 方法 。 

语法 : 

void include(string filename); 

参数 filename 是 指定 的 完整 路 径 文件 名 。 


et 


例 2.38 如果 Web 页 面具 有 一 致 的 外 观 ， 可 以 在 PHP 中 使 用 include 语句 将 模板 和 标准 元 素 载 入 到 页 
面 中 。 下 面 讲解 应 用 include 语句 引用 外 部 文件 的 方法 。《〈 实 例 位 置 : 光盘 \TMNInstance\02\2.38) 
(1) 首先 创建 一 个 简单 的 PHP 动态 页 ， 命 名 为 ndex.php。 
(2) 然后 应 用 include 语句 嵌入 3 个 外 部 文件 ， 分 别 是 top.php 页 、main .php 页 和 bottom.php 页 。 代 码 
如 下 : 
<table width="975" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td><?php include("top.php");?></td> 
</tr> 


<tr> 
<td><?php include("main.php");?></td> 

</tr> 

<tr> 
<td><?php include("bottom.php");?></td> 

</tr> 

</table> 
(3) 最 后 在 以 上 3 个 外 部 文件 中 分 别 调用 图 片 资源 及 数据 信 


息 。 运 行 结果 如 图 2.22 所 示 。 


2.8.2 ”应 用 require 语句 引用 文件 


我 是 引信 的 是 部 


require 语句 的 使 用 方法 与 include 语句 类 似 ， 都 是 实现 对 外 部 “| 
文件 的 引用 。 在 PHP 文件 被 执行 前 ，PHP 解析 器 会 用 被 引用 的 文 Ce 
件 的 全 部 内 容 蔡 换 require 语句 , 然后 与 require 语句 之 外 的 其 他 语 图 2.22 应 用 include 语句 引用 文件 
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句 组 成 新 的 PHP 文件 ， 最 后 再 按 新 PHP 文件 执行 程序 代码 。 


和 5 广 音 因为 require 语句 相当 于 将 另 一 个 源 文件 的 内 容 完全 复制 到 本 文件 中 ， 所 以 一 般 将 其 放 在 源 文 
件 的 起 始 位 置 ， 用 于 引用 需要 使 用 的 公共 函数 文件 和 公共 类 文件 等 。 


PHP 可 以 使 用 任何 扩展 名 来 命名 引用 文件 ， 如 .ine 文件 、.html 文件 或 其 他 非 标准 的 扩展 名 文件 等 ， 但 
PHP 通常 用 来 解析 扩展 名 被 定义 为 .php 的 文件 。 建 议 读者 使 用 标准 的 文件 扩展 名 。 

语法 : 

void require(string filename); 

参数 flename 是 指定 的 完整 路 径 文件 名 。 

例 2.39 应 用 require 语句 引用 并 运行 指定 的 外 部 文件 top.php。〈 实 例 位 置 : 光盘 \TMNInstance\02\2.39) 

代码 如 下 : 

<?php 

require("top.php"); /| 嵌入 外 部 文件 top.php 页 

?> 


top.php 文件 的 代码 如 下 : 
<table width="750" height="131" border="0" cellpadding="0" cellspacing="0"> 

<tr> 

<td bgcolor="#FFCCCC" align="center> 明日 科技 有 限 公司 是 一 家 知名 的 软件 公司 ， 公 司 的 网 址 是 

www.mingribook.com</td> 

</tr> 
</table> 
运行 index.php 页 ,输出 显示 了 require 语句 引用 的 Web [ee ee 

页 。 运行 结果 如 图 2.23 所 示 。 ae wotons va 


2.8.3 应 用 include_once 语句 引用 文件 


室 4 路 FrequrelB- 葵 " 园 -了 师 ”FEPv 


明日 科技 有 限 公司 是 一 家 知名 的 软件 公司 ， 公 司 的 网 址 是 www, ningribook com 


在 使 用 include_once 语句 时 ， 应 该 明确 它 与 include 语 [ACE wa- 
句 的 区 别 ， 应 用 include_once 语句 会 在 导入 文件 前 先 检测 2.23 ”应 用 require 语句 引用 外 部 文件 
该 文件 是 否 在 该 页 面 的 其 他 部 分 被 引用 过 ， 如 果 有 ， 则 不 
会 重复 引用 该 文件 ， 程 序 只 能 引用 一 次 。 例 如 ， 要 导入 的 文件 中 存在 一 些 自 定义 函数 ， 那 么 如 果 在 同一 个 
程序 中 重复 导入 这 个 文件 ， 在 第 二 次 导入 时 便 会 发 生 错 误 ， 因 为 PHP 不 允许 相同 名 称 的 函数 被 重复 声明 。 
语法 : 
Void include_once (string filename); 
参数 filename 是 指定 的 完整 路 径 文件 名 。 
例 2.40 应 用 include_once 语句 引用 并 运行 指定 的 外 部 文件 tppphp。( 实 例 位 置 :光盘 \TMNInstance\02\2.40) 
代码 如 下 : 
<?php 
include_once("top.php"); /| 嵌入 外 部 文件 top.php 页 
?> 
top.php 文件 的 代码 如 下 : 
<table width="779" height="80" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td bgcolor=#33CCFF"> 使 用 include_once 语句 引用 的 文件 </td> 
</tr> 
</table> 
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运行 结果 如 图 2.24 所 示 。 
2.8.4 应 用 require_once 语句 引用 文件 


Tequire once 语句 是 require 语句 的 延伸 , 它 的 功能 与 require 语句 基本 类 似 , 不 同 的 是 ,在 应 用 require_once 
语句 时 会 先 检查 要 引用 的 文件 是 不 是 已 经 在 该 程序 中 的 其 他 地 方 被 引用 过 ， 如 果 有 ， 则 不 会 再 次 重复 调用 
该 文件 。 例 如 ， 同 时 应 用 require_once 语句 在 同一 页 面 中 引用 了 两 个 相同 的 文件 ， 那 么 在 输出 时 只 有 第 一 个 
文件 被 执行 ， 第 二 次 引用 的 文件 不 会 被 执行 。 

语法 : 

Void require_once (string filename); 

参数 filename 是 指定 的 完整 路 径 文件 名 。 

例 2.41 下 面 讲解 应 用 require_once 语句 调用 外 部 文件 的 实现 方法 。( 实 例 位 置 : 光盘 \TMNInstance\02\2.41) 

(1) 首先 创建 一 个 简单 php 页 ， 命 名 为 index.php， 然 后 应 用 require_once 语句 嵌入 3 个 外 部 文件 。 代 
码 如 下 : 
<table width="1004" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td><?php require_once("top.php");?></td> /| 嵌入 头 文件 
</tr> 
<tr> 
<td><?php require_once("main.php");?></td> / 峙 入 主 文件 
</tr> 
<tr> 
<td><?php require_once("bottom.php");?></td> /| 嵌入 尾 文件 
</tr> 
</table> 


本地 rtanet | 保 PP 稚 志保 用 


2.24 ”应 用 include_once 语句 引用 文件 2.25 应 用 require_once 语句 引用 文件 
2.8.5 include 语句 和 require 语句 的 区 别 


应 用 require 语句 来 调用 文件 ， 其 应 用 方法 和 include 语句 是 类 似 的 ， 但 存在 如 下 区 别 。 

在 使 用 require 语句 调用 文件 时 ， 如 果 调 用 的 文件 没 找到 ，require 语句 会 输出 错误 信息 ， 并 且 立 即 
终止 脚本 的 处 理 。 而 include 语句 在 没有 找到 文件 时 则 会 输出 警告 ， 不 会 终止 脚本 的 处 理 。 

使 用 require 语句 调用 文件 时 ， 只 要 程序 一 执行 ， 就 会 立刻 调用 外 部 文件 ; 而 通过 include 语句 调用 


® 
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外 部 文件 时 ， 只 有 程序 执行 到 该 语句 时 ， 才 会 调用 外 部 文件 。 
2.8.6 include_once 语句 和 require_once 语句 的 区 别 


include_once 语句 和 require_once 语句 的 用 途 是 确保 一 个 被 包含 文件 只 能 被 包含 一 次 。 使 用 这 两 个 语句 
可 以 防止 意外 地 多 次 包含 相同 的 函数 库 ， 从 而 减少 函数 重复 定义 并 产生 错误 。 

但 两 者 之 间 是 有 区 别 的 ，include_once 语句 在 脚本 执行 期 间 调 用 外 部 文件 发 生 错误 时 ， 产 生 一 个 警告 ， 
而 require_once 语句 则 导致 一 个 致命 错误 。 


2.9 实 战 
殴 il 视频 讲解 :光盘 \TM\Video\ 第 2 章 \ 实 战 .exe 


2.9.1 判断 半年 的 方法 


例 2.42 下 面 应 用 运算 符 和 表达 式 来 判断 用 户 指定 的 年 份 是 否 是 头 年 。 半 年 的 判断 方式 是 ， 能 够 被 4 
整除 不 能 够 被 100 整除 ， 或 者 能 够 被 100 整除 并 且 能 被 400 整除 的 年 份 为 闽 年 。( 实 例 位 置 ， 光盘 \TM 
Instance\02\2.42) 


本 实例 的 完整 代码 如 下 : 
<table width="392" height="101" border="1" align="center cellpadding="1" cellspacing="1" bordercolor="#009900" 
bgcolor='"#CCFF00"> 

<tr> 

<td height="25" colspan="2" align="center" class="STYLE5"> 判 断 指定 的 年 份 是 否 为 闽 年 </td> 

</tr> 


<tr> 
<td width="160" height="30" bgcolor="#FFFF99"><span class="STYLE3"> 请 输入 一 个 四 位 数 的 年 份 : 
</span></td> 
<td width="219" bgcolor="#FFFF99"> 
<form name="form1" method="post" action="> 
<table width="196" height="29" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="136"><input name="txt_year" type="text" value="" size="20"> </td> 
<td width="60">&nbsp; 
<input type="submit" name="Button" value=" 计 算 "></td> 
</tr> 
</table> 
</form> 
</td> 


<td height="37" colspan="2" align="center"> 
<?php 
if($_POST[Button]!="){ 
S$year=$_POSTI[txt_year]; 
Sresult =($year%4==0 && $year%100!=0)||($year%400==0)?$year." 是 六 年 "-$year." 不 是 半年 "; 
echo $result; 
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> 
</td> 
</tr> 
</table> 


运行 本 实例 ， 在 文本 框 中 输入 指定 的 年 份 ， 单 击 “ 计 算 ” 按 钮 ， 即 可 求 出 该 年 份 是 否 为 闵 年 。 运 行 结 
果 如 图 2.26 所 示 。 


2.9.2 通过 自 定义 函数 防止 新 闻 主 题 信息 出 现 中文 乱 码 


例 2.43 ”创建 一 个 自 定义 函数 ,用 于 实现 屏蔽 中 文 乱码 的 输出 ， 将 该 函数 封装 在 一 个 function.php 文件 
中 ， 然 后 应 用 include_once 语句 引用 这 个 文件 ， 再 通过 echo 语句 输出 新 闻 主题 信息 ， 截 取 前 18 个 字符 ， 并 
应 用 自 定义 函数 屏蔽 中 文 乱码 。 (实例 位 置 : 光盘 \TMNInstance\02\2.43) 

(1) 为 了 保持 整个 页 面 的 合理 布局 ， 经 常 需要 对 一 些 较 长 的 字符 进行 部 分 输出 ， 但 由 于 汉字 占有 两 个 
字符 ， 如 果 截 取 位 置 选取 不 当 就 可 能 导致 截取 的 字符 串 末 尾 出 现 乱 码 ， 本 例 将 解决 上 述 问题 。 

运行 本 实例 ， 结 果 如 图 2.27 所 示 ， 无 论 在 何 处 对 字符 串 进行 截取 都 不 会 出 现 乱码 。 
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放 卫 二 的 后 是 可 广 国 于 


请 六 入 一 个 下 位 才 的 年 内 :| 


0 不 量 加 年 


不 地 Intranet | 作 } 闪 :禁用 
图 2.26 判断 指定 的 年 份 是 否 为 半年 图 2.27 新 闻 主题 信息 显示 


(2) PHP 对 字符 串 进行 截取 可 以 采用 PHP 的 预定 义 函数 substr0 实 现 。 该 函数 的 使 用 方法 如 下 : 
string substr ( string string, int start [, int length]) 


pA 
该 函数 对 string 进行 截取 操作 ， 截 取 从 字符 束 的 start 位 置 开始 ， 到 length 长 度 结 来. 


应 用 substr0 函 数 对 普通 字符 进行 截取 操作 会 非常 方便 , 但 对 全 角 字 符 进 行 截取 时 可 能 会 造成 乱码 出 现 。 
为 了 解决 该 问题 可 以 通过 自 定义 函数 chinesesubstr0 解 决 上 述 问 题 。 该 函数 代码 如 下 : 
<?php 
function chinesesubstr($str, $start, $len) { 
$strlen=$start+$len; 
for($i=0;$i<$strlen;$i++){ 
if(ord(substr($str,$i,1))>0xa0) { 
S$tmpstr.=substr($str, $i,2); 
$i++; 
}else 
S$tmpstr.=substr($str, $i,1); 


} 
return $tmpstr; 


?> 


该 函数 中 的 参数 $str 表示 要 截取 的 字符 串 ，$start 表示 从 何 位 置 开始 截取 ，Slen 表示 截取 字符 串 的 长 度 ， 
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考虑 到 编程 人 员 的 习惯 , 自 定义 函数 chinesesubstr0 与 PHP 预定 义 函 数 substr0 的 参数 设计 一 致 , 这样 程序 员 
就 可 以 像 使 用 函数 substr0 一 样 使 用 chinesesubstr0 了 。 这 里 建议 将 chinesesubstr0 函 数 单独 放 入 某 文件 , 使 用 
时 用 include0 函 数 包 含 该 文件 即 可 。 
(3) 引 用 自 定义 函数 chinesesubstr0 所 在 的 function.php 文件 , 并 通过 循环 显示 公告 主题 的 前 16 个 字符 。 
<table width="160" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="165" height="32"><img src="images/tell top.gif' width="165" height="32" border="0" ></td> 
</tr> 
<tr> 
<td height="52" background="images/tell_center.gif’> 
<?php 
include_once("function.php"); 
$news=" 明 日 科技 编程 词典 !"; 
$i=1; 


?> 
<table width="148" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 


<td width="17" height="20"><img src="images/mark_0.gif" width="10" height="10"></td> 
<td width="333"> 
<?php 
echo chinesesubstr($news,0,18); 
if(stren(S$news)>18X{ 
ee 


} 
?> 
</td> 
</tr> 
<tr> 
<td height="5"></td> 
<td height="5" background="images/back_point_write.gif’></td> 
</tr> 
</table> 
<?php 
$i++; 
}while($i<=5); 
?> 
</td> 
</tr> 
<tr> 
<td width="165" height="12"><img src="images/tell_bottom.gif" width="165" height="12"></td> 
</tr> 
</table> 


By 
> 日 
A 在 PHP 动态 网 页 开发 的 过 程 中 ， 从 数据 库 读 取 数 据 最 为 常用 ， 由 于 数据 库 的 内 容 在 后 面 的 章 
节 中 进行 讲解 ， 因 此 ， 本 例 使 用 了 字符 串 变 量 替代 数据 库 数 据 ， 在 数据 库 章节 中 会 讲解 如 何 读 取 数 据 库 
中 的 内 容 。 
2.9.3 应 用 include 语句 构建 在 线 音乐 网 站 主页 


例 2.44 本 实例 应 用 include 语句 引用 外 部 文件 来 构建 网 站 主页 。 要 使 网 站 主页 的 代码 简洁 化 ， 而 且 维 
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护 起 来 很 方便 ， 只 需要 在 主页 应 用 表格 制作 一 个 简练 、 大 气 、 个 性 鲜明 的 网 页 布局 ， 然 后 将 每 个 布局 区 域 
内 的 内 容 封 装 成 一 个 独立 的 文件 ， 最 后 应 用 include 语句 引用 这 些 独 立 的 文件 即 可 轻松 构建 主页 。 下 面 讲解 
应 用 include 语句 引用 外 部 文件 的 方法 。 (实例 位 置 : 光盘 \TM\Instance\02\2.44) 

对 于 一 个 网 站 而 言 ， 主 页 作为 直接 的 信息 展台 ， 所 承载 的 信息 量 将 是 巨大 的 。 从 网 站 程序 编写 来 说 ， 
编写 首页 的 代码 需要 几 百 行 ， 甚 至 上 千 行 ， 这 对 于 后 期 维护 来 说 是 很 麻烦 的 ， 尤 其 是 当 其 他 管理 员 进 行 维 
护 时 ， 由 于 对 代码 不 熟悉 ， 经 常 需要 在 几 千 行 代码 中 来 回 查找 。 那 么 ， 如 何 来 解决 这 种 问题 呢 ? 应 用 引用 
文件 是 解决 该 问题 最 有 效 的 方法 。 将 拥有 指定 功能 的 代码 封装 成 一 个 单独 的 文件 ， 当 用 到 该 文件 时 ， 只 需 
要 使 用 include 等 引用 语句 对 该 文件 进行 引用 即 可 。 这 样 做 的 目的 在 于 使 得 对 页 面 的 管理 和 维护 都 变 得 更 简 
洁 、 方 便 ， 并 且 减 少 了 代码 的 重用 性 。 

本 实例 的 实现 方法 如 下 : 

(1) 首先 创建 一 个 简单 PHP 动态 页 ， 命 名 为 index.php。 
(2) 设计 一 个 3 行 1 列 的 表格 ,并 将 第 二 行 拆 分 成 两 个 单元 格 。 然 后 应 用 include 语句 引用 4 个 外 部 文 
件 ， 分 别 是 top.html 页 、left.html 页 、main.html 页 和 bottom.html 页 。 构 建 网 站 主页 布局 的 效果 如 图 2.28 所 示 。 
在 index.php 文件 中 应 用 include 语句 引用 4 个 外 部 文件 的 代码 如 下 : 
<table width="768" border="0" align="center" cell padding="0" cellspacing="0"> 
<tr> 
<td colspan="2"><?php include("top.html");?></td> 
</tr> 
<tr> 
<td width="180"><?php include("left.html"); ?></td> 
<td width="588"><?php include("main.html"); ?></td> 
<td colspan="2"><?php incluce("bottom.html");?></td> 
</tr> 


</table> 
(3) 在 引用 的 其 他 子 页 中 应 用 HTML 标记 设计 子 页 内 容 。 运 行 结果 如 图 2.29 所 示 。 
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/ 
四 在 PHP 动态 网 页 开发 的 过 程 中 ， 在 功能 上 使 用 最 频繁 的 就 是 对 数据 库 连 接 文件 的 引用 。 将 数 
据 库 连接 文件 封装 在 一 个 独立 的 文件 中 ， 然 后 应 用 include 语句 或 者 require 语句 等 对 封装 的 文件 进行 引 
用 ， 由 于 数据 库 内 容 将 在 后 面 的 章节 中 进行 介绍 ， 因 此 ， 引 用 数据 库 文件 的 方法 将 在 后 面 进行 详细 介绍 。 


2.9.4 随机 组 合生 日 祝福 语 


例 2.45 创建 index.php 文件 ， 首 先 定义 两 个 数组 ， 将 字符 串 信 息 保存 在 数组 中 。 然 后 ， 利 用 rand 函数 
随机 取得 数组 中 的 两 条 信息 并 保存 在 变量 中 。 (实例 位 置 : 光盘 \TMVInstance\02\2.45) 
本 实例 的 完整 代码 如 下 : 
<?php 
$arr = array(" 生 日 快乐 "," 今 天 是 你 的 出 生日 "," 同 学 们 为 你 许愿 "); 
$arr1 = array(" 祝 你 万 事 如 意 "," 祝 你 生日 快乐 "," 祝 你 每 天 都 有 一 份 好 心情 "); 
Srand = rand(0,2); 
echo $arr[$rand].$arr1[$rand]; 


?> 
运行 结果 如 图 2.30 所 示 。 


2.9.5 ”加 法 计算 器 


例 2.46 下 面 制作 一 个 简单 的 加 数 器 ， 当 用 户 单 击 “ 等 于 ”按钮 时 ， 通 过 算术 运算 符 “+” 计 算 两 个 文 
本 框 中 数据 的 和 ， 并 显示 在 文本 框 中 。《〈 实 例 位置 : 光盘 \TMNInstance\02\2.46) 

(1) 首先 ， 创 建 index.php 文件 ， 设 计 表 单 页 面 ， 效 果 如 图 2.31 所 示 。 

(2) 获取 表单 中 提交 的 数据 ， 完 成 加 法 运算 。 代 码 如 下 : 


<?phl 
issetS_PosTrsubmit) and $_POST[number1]i=null and $_POST[number2"]!=null and $_POST[Submit]==" 
等 于 "X( 
$number3=$_POST[number1"]+$_POST['number2; /计算 变量 值 
jelsef 
$number3=null; 
1 
?> 
<label> 
<input name="number3" type="text" id="number3" size="10" value="<?php echo $number3;?>"/> <!-- 输 出 变 
量 值 -> 
</label> 
运行 结果 如 图 2.31 所 示 。 


EH Windows rere Epiorer 


Exple cn 
cl EE 


RMD MBE) EV) XIA a 有 加 法 计数 器 


区 


| 生日 尿 各 如 万 各 写 3 + i 0 
ET Tr = ETT 
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2.10 小 结 


本 章 主要 介绍 了 PHP 语言 的 基础 知识 ， 重 点 讲解 了 PHP 的 语法 基础 、 数 据 类 型 、 常 量 、 变 量 、 运 算 符 
和 表达 式 ， 以 及 PHP 函数 、 输 出 语句 等 。 基 础 知识 是 一 门 语言 的 核心 ， 希 望 初学 者 能 静 下 心 来 ， 牢 牢 掌握 
本 章 的 知识 。 只 有 基础 扎实 ， 才 能 真正 迈 过 精通 PHP 的 门槛 。 


2.11 学 习 成 果 检 验 


图 2.32 字符 串 运算 符 的 应 用 


2. 在 开发 动态 网 站 中 的 用 户 注册 模块 时 ， 其 中 用 户 注册 的 密码 一 般 都 要 填写 一 个 确认 密码 ， 来 加 强 输 
入 信息 的 准确 性 。 下 面 在 用 户 注册 模块 中 ， 应 用 比较 运算 符 验证 两 次 输入 的 密码 是 否 一 致 。 如 果 相等 则 输 
出 注册 成 功 信息 ， 和 否则 将 输出 错误 提示 信息 。 (答案 位 置 ， 光盘 \TM\Instance\02\2.48) 

3， PHP 的 输出 语句 有 echo、print、printf、print r。 尝 试 使 用 这 4 个 语句 分 别 输出 数据 ， 看 它们 之 间 有 
什么 不 同 。 (答案 位 置 ， 光盘 \TM\Instance\02\2.49) 
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(如 视频 讲解 : 54 分钟 ) 


条 件 控制 语句 和 循环 控制 语句 是 两 种 基本 的 语法 结构 ， 它 们 都 是 
用 来 控制 程序 执行 流程 的 ， 也 是 构成 程序 的 主要 语法 基础 。 通 常 ， 语 
句 是 按 顺序 执行 的 ， 即 执行 完 当前 的 语句 后 ， 执 行 下 一 条 语句 。 但 是 
这 种 简单 的 顺序 执行 方式 能 够 解决 的 问题 很 有 有限， 现实 生活 中 的 许多 
逻辑 使 用 顺序 执行 的 方式 是 不 能 够 被 描述 的 。 因 此 ， 开 发 人 员 可 以 按 
照 实际 问题 的 逻辑 思路 选择 性 地 使 用 流程 控制 语句 来 编写 程序 代码 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

Wp 了 解 IF 语句 的 使 用 

Nm 了 解 switch…case 多 重 判断 语句 

Wm 掌握 while 循环 语句 

MH 掌握 do…while 循环 语句 

mi 掌握 for 循环 语句 

MH 掌握 foreach 循环 语句 

mm 了 解 break/continue 关键 字 
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3.1 控制 结构 


在 编程 的 过 程 中 ， 所 有 的 操作 都 是 在 按照 某 种 结构 有 条 不 率 地 进行 ， 纠 其 根源 ， 程 序 的 控制 结构 大 致 
可 以 分 为 3 种 : 顺序 结构 、 选 择 结构 和 循环 结构 。 在 对 这 3 种 结构 的 使 用 中 ， 几 乎 很 少 有 哪个 程序 是 单独 
使 用 某 一 种 结构 来 完成 某 个 操作 的 ， 基 本 上 都 是 其 中 的 两 种 或 者 3 种 结构 结合 使 用 。 为 了 帮助 读者 更 好 地 
应 用 这 3 种 程序 控制 结构 ， 下 面 对 其 进行 详细 介绍 。 

1. 顺序 结构 

顺序 结构 是 最 简单 、 最 基本 的 结构 方式 ， 各 流程 框 依次 按 顺 序 执行 。 其 传统 流程 图 的 表示 方式 与 N-S 
结构 化 流程 图 的 表示 方式 分 别 如 图 3.1 和 图 3.2 所 示 。 其 执行 顺序 为 ， 开始 一 语句 1 一 语句 2 一 … 一 结束 。 


语 名 1 
语 句 2 
图 3.1 顺序 结构 传统 流程 图 图 3.2 N-S 结构 化 流程 图 


2. 选择 〈 分 支 ) 结构 
选择 结构 就 是 对 给 定 条件 进 行 判断 ， 条 件 为 真 时 执行 一 个 分 支 ， 条 件 为 假 时 执行 另 一 个 分 支 。 其 传统 
流程 图 表示 方式 与 N-S 结构 化 流程 图 表示 方式 分 别 如 图 3.3 和 图 3.4 所 示 。 


< 和 件 一 > ! 
0 
OO 
传统 流程 图 
图 3.3 条 件 成 立 与 否 都 执行 语句 或 语句 块 


| 


语句 块 1 


了 5 结构 化 流程 图 


RS 结构 化 流程 图 


传统 流程 图 
3.4 条 件 为 否则 不 执行 语句 或 语句 块 


3. 循环 结构 
循环 结构 可 以 根据 需要 多 次 重复 执行 一 行 或 者 多 行 代码 。 循 环 结构 分 为 两 种 ， 前 测试 型 循环 和 后 测试 


® 
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型 循环 。 

前 测试 型 循环 ， 先 判断 后 执行 。 当 条 件 为 真 时 反复 执行 语句 或 语句 块 ， 条 件 为 假 时 ， 跳 出 循环 ， 继 续 
执行 循环 后 面 的 语句 ， 流 程 图 如 图 3.5 所 示 。 

后 测试 型 循环 ， 先 执行 后 判断 。 先 执行 语句 或 语句 块 ， 再 进行 条 件 判断 ， 直 到 条 件 为 假 时 ， 跳 出 循环 ， 
继续 执行 循环 后 面 的 语句 ， 否 则 一 直 执行 语句 或 语句 块 ， 流 程 图 如 图 3.6 所 示 。 


当 条 件 成 立 
语句 块 
语句 块 
直到 条 件 不 成 立 
传统 流程 图 了 -结构 化 流程 传统 流程 图 IF 3 结构 化 流程 图 
图 3.5 前 测试 型 循环 流程 图 图 3.6 后 测试 型 循环 流程 图 
在 PHP 中 ， 大 多 数 情况 下 程序 不 会 是 简单 的 顺序 结构 ， 而 是 以 这 3 种 结构 的 组 合 形式 出 现 。 其 中 的 顺 
序 结构 很 容易 理解 ， 就 是 直接 输出 程序 运行 结果 ， 而 选择 和 循环 结构 则 需要 以 下 3 种 控制 语句 来 实现 。 


条 件 控 制 语句 过、else、elseif 和 switch 语句 。 

循环 控制 语句 :while、do…while、for 和 foreach 语句 。 
跳 转 控制 语句 : break 和 continue 语句 。 

在 下 面 的 章节 中 将 对 这 3 种 控制 语句 进行 详细 讲解 。 


3.2 条件 控 制 语句 


所 谓 条 件 控制 语句 就 是 对 语句 中 不 同 条 件 的 值 进 行 判断 ， 进 而 根据 不 同 的 条 件 执行 不 同 的 语句 。 在 条 
件 控制 语句 中 主要 有 两 个 语句 : 让 条 件 控制 语句 和 switch 多 分 支 语句 。 


3.2.1 j 半 条 件 控制 语句 


铭 i 视频 讲解 : 光盘 \TMVVideo\ 第 3 章 \if 条 件 控制 语句 .exe 

1. if 语 句 

让 语句 是 最 简单 的 条 件 判 定语 句 , 它 对 某 段 程序 的 执行 附加 一 个 条 件 , 如 果 条 件 成 立 , 就 执行 这 段 程序 ; 
否则 就 跳 过 这 段 程序 去 执行 下 面 的 程序 。 

让 语句 的 格式 为 : 

if (expr) 

statement ; 

如 果 表达 式 expr 的 值 为 me， 那么 就 顺序 执行 statement 语句 ;和 否则， 就 跳 过 该 条 语句 再 往 下 执行 。 如 

果 需 要 执行 的 语句 不 只 一 条 ， 那 么 可 以 使 用 “{ }” (在 “{}” 中 的 语句 ， 被 称 之 为 语句 组 ) 。 其 格式 为 : 


if(expr}{ 
statement1; 
statement2; 
} 


局 
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让 条 件 语 句 的 流程 图 如 图 3.7 所 示 。 
例 3.1 在 本 例 中 ， 实 现 输出 “如 果 元 旦 放假 ， 我 们 一 起 去 息 山 ”。 首 先 为 变量 赋 一 个 逻辑 值 ， 然 后 判 
断 这 个 值 是 否 为 真 ， 如 果 为 真 ， 则 输出 结果 。 实例 位 置 ， 光盘 \TM\Instance\03\3.1) 


代码 如 下 : 
<?php 
$day_51=true; // 为 变量 赋予 一 个 逻辑 值 
if ($day_51==trueX{ // 判 断 变 量 的 逻辑 值 是 否 为 真 
echo "如 果 元 旦 放假 ， 我 们 一 起 去 扑 山 "; // 输 出 字符 串 
?> l 
运行 结果 如 图 3.8 所 示 。 


去 bs 下 -| 而 记 | 公 co- 


加 昌 元 旦 耻 本 ,起 们 一 起 去 区 山 
二 证 取 Jntrener | pu 介 用 


图 3.7 让 语句 流程 图 图 3.8 让 语句 的 应 用 
2. if…else 语句 
大 多 数 时 候 ， 总 是 需要 在 满足 某 个 条 件 时 执行 一 条 语句 ， 而 在 不 满足 该 条 件 时 执行 其 他 语句 。 为 了 在 
站 语句 中 描述 这 种 情况 ， 直 语句 中 提供 了 else 子 句 ，else 子 句 表示 的 自然 语句 是 “否则 ”的 意思 。 其 语法 格 
式 为 : 
ifexprj{ 
statement1; 


Jelse{ 
statement2; 


} 

当 表 达 式 expr 的 值 为 tue 时 ， 执 行 statementl 语句 :如果 表达 式 expr 的 值 为 false， 则 执行 statement2 
语句 。if…else 语句 的 流程 图 如 图 3.9 所 示 。 

例 3.2 在 许多 问题 的 逻辑 关系 中 ， 经 常会 出 现 依据 一 个 条 件 是 否 成 立会 导致 两 种 不 同 的 结果 。 例 如 ， 
如 果 元 旦 放假 ， 我 们 一 起 去 爬山 ， 否 则 我 们 在 家 看 电视 。 首 先 为 变量 赋 一 个 逻辑 值 ， 然 后 判断 这 个 值 是 否 
为 tme， 如 果 为 tue， 则 输出 “如 果 元 旦 放假 ， 我 们 一 起 去 肘 山 ”否则 输出 “我 们 在 家 看 电视 ”， 根 据 不 
同 的 结果 显示 不 同 的 字符 串 。 (实例 位 置 : 光盘 \TM\Instance\03\3.2) 


实例 代码 如 下 : 
<?php 
$day_51=false; /为 变量 赋予 一 个 逻辑 值 
if ($day_51==trueX{ // 判 断 变 量 的 逻辑 值 是 否 为 真 
echo "如 果 元 旦 放假 ， 我 们 一 起 去 胞 山 "; // 输 出 字符 串 
} 
el: 
echo "我 们 在 家 看 电视 "; // 输 出 字符 串 
Rs 
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结果 为 : 我 们 在 家 看 电视 。 
3. 赃 套 的 else if 结构 语句 
使 用 让 语句 和 else 子 句 能 够 描述 一 些 复杂 的 逻辑 问题 ， 但 是 有 时 并 不 能 够 完整 地 表达 人 们 的 语义 。 考 
虑 这 样 一 种 情况 ，if…else 语句 只 能 选择 两 种 结果 : 要 么 执行 真 ， 要 么 执行 假 。 但 如 果 现 在 有 两 种 以 上 的 选 
择 该 怎么 办 呢 ? 例如 ， 小 红 的 考试 成 绩 ， 如 果 是 90 分 以 上 ， 则 为 “优秀 ”; 如 果 是 60 一 90 分 之 间 ， 则 为 
“良好 ”; 如果 低 于 60 分 ， 则 为 “不 及 格 ”。 这 时 ， 可 以 使 用 else if (也 可 以 写 做 else if) 语句 来 执行 。 该 
语法 格式 为 : 
if(expr1X{ 
statement 1; 
}else if(expr2X{ 
statement 2; 
二 
else{ 
statement n; 


} 
else 站 语句 的 流程 图 如 图 3.10 所 示 。 


3.9 ”if…else 语句 流程 图 3.10 ”else 站 语句 的 流程 图 


例 3.3 通过 else 让 语句 ， 判 断 小 红 的 考试 情况 。 如 果 成 绩 大 于 等 于 90， 为 优秀 ， 如 果 成 绩 大 于 等 于 
80， 则 为 良好 ;如 果 成 绩 大 于 等 于 60， 则 为 及 格 ， 否 则 为 不 及 格 。〔 实 例 位 置 ， 光盘 \TMNInstance\03\3.3) 
实例 代码 如 下 : 


<?php 
$score=95; // 设 置 小 红 期 未 考试 的 默认 值 
if ($score >=90X{ // 判 断 小 红 的 期 末 考 试 成 绩 是 否 在 90 分 以 上 
echo "小 红 期 末 考 试 成 绩优 秀 "; /如 果 是 ， 说 明 小 红 期 未 考试 成 绩优 秀 
jelseif($score<90 && $score>=80X{ /1 否则 判断 小 红 期 末 考 试 成 绩 是 否 在 80 一 90 之 间 
echo "小 红 期 末 考 试 成 绩 良 好 "; /如 果 是 ， 说 明 小 红 期 未 考试 成 绩 良好 
jelseif($score<80 && $score>=60}{ // 否 则 判断 小 红 期 未 考试 成 绩 是否 在 60~80 之 间 
echo "小 红 期 末 考 试 成 绩 及 格 "; /如 果 是 ， 说 明 小 红 期 未 考试 成 绩 及 格 
Jelse{ // 如 果 两 个 判断 都 是 false， 则 输出 默认 值 
echo "小 红 期 末 考 试 成 绩 不 及 格 "; /说 明 小 红 期 末 考 试 成 绩 不 及 格 
} 
2 


结果 为 小 红 期 未 考试 成 绩优 秀 。 


@_ 
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人 注意 让 语句 和 else 证 语句 的 执行 条 件 是 表达 式 的 值 为 真 ， 而 else 执行 条 件 是 表达 式 的 值 为 假 。 这 里 
表达 式 的 值 不 等 于 变量 的 值 。 


3.2.2 switch…case 分 支 控制 语句 


区 il 视频 讲解 : 光盘 \TM\Video\ 第 3 章 \switch…case 分 支 控制 语句 .exe 
在 程序 设计 中 ， 所 有 依据 条 件 作出 判定 的 问题 ， 都 可 以 用 前 面 所 介绍 的 不 同类 型 的 计 语句 来 解决 。 不 
过 , 在 用 站 …else 语句 处 理 多 个 条 件 的 判定 问题 时 , 组 成 条 件 的 表达 式 在 每 一 个 else 站 语 句 中 都 要 计算 一 次 ， 
显得 繁琐 腔 肿 ,为 了 避免 站 语句 的 元 长 ,提高 程序 的 可 读 性 , 可 以 使 用 switch 分 支 控制 语句 .PHP 提供 switch… 
case 多 分 支 控制 语句 ， 对 于 某 些 多 项 选择 场合 ， 会 使 程序 代码 更 加 简洁 、 易 读 。 
switch 语句 的 语法 格式 如 下 : 
switch(variable){ 
case value1: 
statement1; 
break; 
case value2: 


default: 
default statement n; 


中 
switch 语句 根据 variable 的 值 ， 依 次 与 case 中 的 value 值 相 比较 ， 如 果 不 相等 ， 继 续 查 找 下 一 个 case; 
如 果 相 等 ， 就 执行 对 应 的 语句 ， 直 到 switch 语句 结束 或 遇 到 break 为 止 。 一般 switch 语句 的 结尾 都 有 一 个 默 
认 值 default， 如 果 在 前 面 的 case 中 没有 找到 相符 合 的 条 件 ， 则 输出 默认 语句 ， 这 和 else 语句 类 似 。 
switch 语句 的 流程 图 如 图 3.11 所 示 。 
例 3.4 将 使 用 switch 语句 来 输出 当天 为 星期 几 , 并 根据 不 同 的 日 期 输出 不 同 的 提醒 警句 。( 实 例 位 置 ; 
光盘 \TMNInstance\03\3.4) 
实例 代码 如 下 : 
<?php 
setlocale(LC_TIME,"chs"); // 设 置 本 地 环境 
$weekday = strftime("%A"); /| 声明 变量 $weekday 的 值 
switch ($weekday}{ //switch 语句 ， 判 断 $weekday 的 值 
case "星期 一 ": // 如 果 变 量 的 值 为 “星期 一 ” 
echo "今天 是 $weekday ， 新 的 一 天 开始 了 ! "; 
break; 
case "星期 二 ": // 如 果 变 量 的 值 为 “星期 二 ” 
echo "今天 是 $weekday ， 认 真 的 工作 态度 真 的 很 重要 !; 
break; 
case "星期 三 ": // 如 果 变 量 的 值 为 “星期 三 ” 
echo "今天 是 $weekday ， 充 实生 活 ， 努 力 工作 !"; 
break; 
case "星期 四 ": // 如 果 变 量 的 值 为 “星期 四 ” 
echo "今天 是 $weekday ， 勤 奋 才 能 创造 绩效 ， 加 油 ! )"; 
break; 
case "星期 五 ": // 如 果 变 量 的 值 为 “星期 五 ” 
echo "今天 是 $weekday ， 积 极 完成 工作 任务 !"; 
break; 


case "星期 六 ": 1/ 如果 变量 的 值 为 “星期 六 ” 


PHP+MySQL 开发 实战 


echo "今天 是 $Sweekday ， 可 以 放松 心情 了 !"; 


break; 
default: / 菊 认 值 
echo "今天 是 $weekday ,去 happy 一 下 ! "; 
break: 
} 
?> 


运行 结果 如 图 3.12 所 示 。 


误 ex* 号 -| 画 : 莹 从 ec- 


天 是 时 其 五 ， 视 忆 元 打工 作 位 骨 ! 
不 地 nvanet| PE 和 用 


图 3.11 switch 语句 流程 图 图 3.12 ”switch*…case 分 支 控制 语句 的 应 用 


NW 


站 switch 语句 在 执行 时 ， 即 使 遇 到 符合 要 求 的 case 语句 段 ， 也 会 继续 往 下 执行 ， 直 到 switch 结 
束 。 为 了 避免 这 种 浪费 时 间 和 资源 的 情况 发 生 ， 一 定 要 在 每 个 case 语句 段 后 添加 break 跳 转 语句 跳出 当 
前 循环 ，break 跳 转 语句 将 在 本 章 3.4.1 节 中 进行 详细 介绍 。 


3.3 ”循环 控制 语句 


循环 语句 是 在 满足 条 件 的 情况 下 反复 执行 某 一 个 操作 。 在 PHP 中 , 提供 4 个 循环 控制 语句 , 分 别 是 while 
循环 语句 、do…while 循环 语句 、for 循环 语句 和 foreach 循环 语句 。 


3.3.1 while 循环 语句 


陋 i 视频 讲解 : 光盘 \TMVVideo\ 第 3 章 \while 循环 语句 .exe 

while 循环 语句 ， 其 作用 是 反复 执行 某 一 项 操作 ， 是 循环 控制 语句 中 最 简单 的 一 个 ,也 是 最 常用 的 一 个 。 
while 循环 语句 对 表达 式 的 值 进行 判断 ， 当 表达 式 为 非 0 值 时 ， 执 行 while 语句 中 的 内 赃 语 句 ， 当 表达 式 的 
值 为 0 值 时 ， 则 不 执行 while 语句 中 的 内 赃 语 句 。 该 语句 的 特点 是 : 先 判断 表达 式 ， 后 执行 语句 。while 循 
环 控制 语句 的 操作 流程 如 图 3.13 所 示 。 


其 语法 如 下 : 
while (expr}{ 是 

statement; 先 判 断 条 件 ， 当 条 件 满足 时 执行 语句 块 ， 否 则 不 向 下 执行 
} WW 


只 要 while 表达 式 expr 的 值 为 tue， 就 重复 执行 嵌 套 中 的 statement 语句 ， 如 果 while 表达 式 的 值 一 开始 


局 
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就 是 false， 则 循环 语句 一 次 也 不 执行 。 
例 3.5 将 10 以 内 的 偶数 输出 。 从 1 一 10 依次 判断 是 否 为 偶数 ， 如 果 是 ， 则 输出 ;如 果 不 是 ， 则 继续 
下 一 次 循环 。《〈 实 例 位置 ， 光盘 \TM\Instance\03\3.5) 
实例 代码 如 下 : 
<?php 
$num = 1; 
$str = "10 以 内 的 偶数 为 :"; 
while($num <= 10X{ 


if($num % 2 == 0X{ 
S$str .= $num." "; 
» 
S$num++; 
5 
echo $str; 


?> 
运行 结果 如 图 3.14 所 示 。 
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3.13 while 循环 控制 语句 的 操作 流程 图 3.14 while 循环 语句 的 应 用 
3.3.2 ”do…while 循环 语句 


如 视频 讲解 : 光盘 \TM\Video\ 第 3 章 \do…while 循环 语句 .exe 

while 语句 还 有 另 一 种 表示 形式 : do…while。do…while 循环 语句 和 while 循环 语句 非常 相似 , 只 是 do… 
while 循环 语句 在 循环 底部 检测 循环 表达 式 ， 而 不 是 在 循环 的 顶部 进行 检测 。 

do…while 循环 语句 的 语法 格式 如 下 : 


statement 
}while(expr); 
do…while 语句 先 执行 statement 语句 , 然后 再 对 表达 式 进行 判断 。 如果 表 达 式 的 值 为 false, 则 跳出 循环 。 
因此 应 用 do…while 循环 语句 时 该 语句 的 循环 体 至 少 被 执行 一 次 。 
do…while 语句 的 流程 控制 图 如 图 3.15 所 示 。 
例 3.6 ”下面 通 过 两 个 语句 的 运行 对 比 来 了 解 两 者 的 不 同 。〔 实 例 位 置 光盘 \TMNInstance\03\3.6) 
实例 代码 如 下 : 
<?php 
$num = 1; 
while($num!=1X{ 
echo "不 会 看 到 "; 
do{ 


四 
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echo "会 看 到 "; 
}while($num!=1); 
2 
结果 为 : 会 看 到 。 
从 上 面 的 代码 中 可 以 看 出 两 者 的 区 别 : do…while 要 比 while 语句 多 循环 一 次 。 当 while 表达 式 的 值 为 假 
时 ，while 循环 直接 跳出 当前 循环 ， 而 do…while 语句 则 是 先 执行 一 遍 程序 块 ， 然 后 再 对 表达 式 进 行 判断 。 


人 注意 do…while 语句 结尾 处 的 while 语句 括号 后 面 有 一 个 分 号 “;”， 为 了 养 成 良好 的 编程 习惯 ， 建 议 
读者 在 书写 的 过 程 中 不 要 将 其 遗漏 。 


3.3.3 for 循环 语句 


芯 视频 讲解 : 光盘 \TM\Video\ 第 3 章 \for 循环 语句 .exe 

for 循环 是 PHP 中 最 复杂 的 循环 结构 。for 循环 语句 能 够 按照 已 知 的 循环 次 数 进行 循环 操作 ， 主 要 应 用 
于 多 条 件 情况 下 的 循环 操作 。 如 果 在 单一 条 件 下 使 用 for 循环 语句 就 有 些 不 合适 。 这 一 点 从 该 语句 的 语法 中 
就 可 以 看 出 ， 其 条 件 的 表达 式 有 3 个 。 它 的 语法 格式 为 : 

for (expr1; expr2; expr3){ 

有 statement; 

其 中 ，exprl 为 变量 初始 赋值 ，expr2 为 循环 条 件 ， 即 在 每 次 循环 开始 前 求 值 。 如 果 值 为 真 ， 则 执行 
statement; 否则 ， 跳 出 循环 ， 继 续 往 下 执行 。expr3 为 变量 递增 或 递减 ， 即 每 次 循环 后 被 执行 。for 循环 语句 
的 流程 图 如 图 3.16 所 示 。 


3.15 do…while 语句 的 控制 流程 图 图 3.16 ”for 循环 语句 的 流程 图 
例 3.7 应 用 for 循环 计算 50 的 阶乘 。《〈 实 例 位 置 : 光盘 \TM\Instance\03\3.7) 
代码 如 下 : 
<?php 
$sum = 1; // 声 明 整 型 变量 $sum 
for ($i = 1:$i ;$i++){ 


// 当 $i<50 时 ， 执 行 该 表达 式 


$sum *= $i; 


} 
echo "50! = ".$sum; // 输 出 该 表达 式 
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结果 为 : 50! = 3.04140932017E+64。 


人 注意 在 for 语 名 中 无 论 采 用 循环 变量 递增 或 递减 的 方式 ， 有 一 个 前 提 ， 就 是 一 定 要 保证 循环 能 够 结 
束 ， 无 期 限 的 循环 ( 死 循环 ) 将 导致 程序 的 崩溃 。 在 循环 变量 自 增 的 例子 中 ， 循 环 的 条 件 是 Si&50， 由 
于 在 每 次 循环 后 $i 的 值 会 加 1， 当 $i 的 值 大 于 50 时 ， 循 环 就 结束 了 . 


上 面 的 代码 采用 了 循环 变量 递增 的 方式 。 当 然 ， 也 可 以 采用 倒序 的 方式 ， 以 循环 变量 递减 的 方式 编写 


程序 。 例 如 : 

<?php 
$sum = 1; // 声 明 整 型 变量 ysum 
for ($i = 50;$i >0:$i- 放 

$sum *= $i; // 当 $i>0 时 ， 执 行 该 表达 式 

} 
echo "50! = ".$sum; // 输 出 该 表达 式 

?> 


结果 为 : 50! = 3.04140932017E+64。 
在 循环 变量 自 减 的 例子 中 ， 循 环 条 件 是 $i>0， 在 每 次 循环 后 $i 的 值 会 减 1， 当 $i<0 时 ， 循 环 就 结束 了 。 


3.3.4 foreach 循环 


foreach 循环 在 PHP 4 引进 的 ， 只 能 用 于 数组 。 在 PHP 5 中 ， 又 增加 了 对 对 象 的 支持 。 该 语句 的 语法 格 
式 为 : 
foreach (array_expression as $value) 
statement 
或 
foreach (array_expression as $key => $value) 
statement 
foreach 语句 将 遍历 数组 array_expression， 每 次 循环 时 ， 将 当前 数组 中 的 值 赋 给 $value (或 是 8key 和 
$value) ， 同 时 ， 数 组 指针 向 后 移动 ， 直 到 遍历 结束 。 当 使 用 foreach 语句 时 ， 数 组 指针 将 自动 被 重 置 ， 所 
以 不 需要 手动 设置 指针 位 置 。 


全 5 注意 foreach 语句 不 支持 用 “@” 来 禁止 错误 信息 。 


例 3.8 foreach 语句 的 应 用 很 广泛 ， 其 主要 功能 就 是 处 理 数 组 ， 下 面 就 应 用 foreach 语句 来 处 理 一 个 数 
组 ， 实 现 输出 购物 车 中 商品 的 功能 。 这 里 假设 将 购物 车 中 的 商品 存储 于 指定 的 数组 中 ， 然 后 通过 foreach 语 
句 来 输出 购物 车 中 的 商品 信息 。《〈 实 例 位置 : 光盘 \ITM\Instance\03\3.8) 

程序 代码 如 下 : 

<?php 

$name = array("1"=>"iPhone5","2"=>" 数 码 相 机 ","3"=>" 联 想 电脑 ","4"=>" 天 王 表 "); 

$price = array("1"=>"79999 元 ","2"=>"3000 元 ","3"=>"5600 元 ","4"=>"3600 元 "); 

$counts = array("1"=>1,"2"=>1,"3"=>2,"4"=>1); 

echo '<table width="580" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" 

bgcolor="#c17e50"> 

<tr> 


<td width="145" align="center" bgcolor="#FFFFFF" class="STYLE1"> 商 品名 称 </td> 

<td width="145" align="center" bgcolor="#FFFFFF” class="STYLE1"> 价 格 </td> 

<td width="145" align="center" bgcolor="#FFFFFF" class="STYLE1"> 数 量 </td> 

<td width="145" align="center" bgcolor="#FFFFFF" class="STYLE1"> 金 额 </td> 
</tr>"; 


@ 
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foreach($name as $key=>S$value){ /以 book 数组 做 循环 ， 输 出 键 和 值 
echo '<tr> 
<td height="25" align="center" bgcolor="#FFFFFF" class="STYLE2">".$value.'</td> 
<td align="center" bgcolor="#FFFFFF" class="STYLE2">".$price[$key].'</td> 
<td align="center" bgcolor="#FFFFFF" class="STYLE2">".$counts[$key].'</td> 
<td align="center" bgcolor="#FFFFFF" class="STYLE2">'.$counts[$key]*$price[$key].'</td> 
</tr>"; 


上 
echo '</table>"; 
?> 


运行 结果 如 图 3.17 所 示 。 


EET 


图 3.17 应 用 foreach 语句 输出 商品 


人 注意 当 试图 使 用 foreach 语句 用 于 其 他 数据 类 型 或 者 未 初始 化 的 变量 时 会 产生 错误 。 为 了 避免 这 个 
问题 ， 最 好 使 用 is_arrayO 函 数 先 来 判断 变量 是 否 为 数组 类 型 ， 如 果 是 ， 再 进行 其 他 操作 。 


3.4” 跳 转 控制 语句 


跳 转 语句 主要 分 为 3 个 部 分 ，break 语句 、continue 语句 和 return 语句 。 其 中 前 两 个 跳 转 语句 使 用 起 来 
非常 简单 而 且 非 常 容 易 掌 握 ， 主 要 原因 是 它们 都 被 应 用 在 指定 的 环境 中 ， 如 for 循环 语句 中 。return 语句 在 
应 用 环境 上 较 前 两 者 相对 单一 。 一 般 被 用 在 自 定义 函数 和 面向 对 象 的 类 中 。 


3.4.1 使 用 break 语句 跳出 循环 


break 关键 字 可 以 终止 当前 的 循环 , 包括 while、do…while 、for、foreach 和 switch 在 内 的 所 有 控制 语句 。 

例 3.9 将 使 用 一 个 while 循环 ，while 后 面 的 条 件 表达 式 的 值 为 tue， 即 为 一 个 无 限 循环 。 在 while 程 
序 块 中 将 声明 一 个 随机 数 变量 gtmp， 只 有 当 生 成 的 随机 数 等 于 10 时 ， 使 用 break 语句 跳出 循环 。 (实例 位 
置 : 光盘 \TM\Instance\03\3.9) 


代码 如 下 : 

<?php 

while(trueX{ /使 用 while 循环 
S$tmp = rand(1,20); /声明 一 个 随机 数 变 量 $Stmp 
echo $tmp." "; /| 输出 随机 数 


局 
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if($tmp == 10X{ // 判 断 随机 数 是 否 等 于 10 
echo "<p> 变 量 等 于 10， 终 止 循环 "; 
break; // 如 果 等 于 10， 使 用 break 语句 跳出 循环 
} 
» 


运行 结果 如 图 3.18 所 示 。 

break 语句 不 仅 可 以 跳出 当前 的 循环 ， 还 可 以 指定 跳出 几 重 循环 。 格 式 为 : 
break n; 

参数 n 指定 要 跳出 的 循环 数量 。break 关键 字 的 流程 图 如 图 3.19 所 示 。 
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下 量 于 10, 加 上 本 


入 林地 intranet | 保护 禾 区 条 用 
图 3.18 ”应 用 break 跳 转 控制 语句 跳出 循环 图 3.19 break 关键 字 的 流程 图 
例 3.10 本 例 共 有 3 层 循环 , 最 外 层 的 while 循环 和 中 间 层 的 for 循环 是 无 限 循 环 , 最 里 面 并 列 两 个 for 
循环 : 程序 首先 执行 第 一 个 for 循环 ， 当 变量 $i 等 于 6 时， 跳出 当前 循环 (一重 循环 ) ， 继 续 执行 第 二 个 for 
循环 ， 当 第 二 个 for 循环 中 的 变量 $j 等 于 10 时 ， 将 直接 跳出 最 外 层 循环 。《〈 实 例 位置 : 光盘 \TM 
Instance\03\3.10) 
实例 代码 如 下 : 


<?php 
while(true){ 
for(;;X{ 
for($i=0;$i<=10;$i++)f 
echo $i." "; 
if($i == 6X{ 
echo "<p> 变 量 \$i 等 于 6， 跳 出 一 重 循环 。<p>"; 
break 1; 
} 
中 
for($j = 0; $j < 20; $j++){ 
echo $j." "; 
if($j == 10X 
echo "<p> 变 量 \$j 等 于 10， 跳 出 最 外 重 循环 。"; 
break 3; 
8 
} 
echo "这 句 话 不 会 被 执行 。"; 
;> 
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运行 结果 如 图 3.20 所 示 。 


3.4.2 ”使 用 continue 语句 跳出 循环 


continue 跳 转 语句 的 作用 没有 break 那么 强大 ， 只 能 终止 本 
次 循环 ， 而 进入 到 下 一 次 循环 中 。 在 执行 continue 语句 后 ， 程 
序 将 结束 本 次 循环 的 执行 ， 并 开始 下 一 轮 循环 的 执行 操作 。 图 3.20 使 用 break 关键 字 跳 出 多 重 循环 
continue 也 可 以 指定 跳出 几 重 循环 。continue 跳 转 语句 的 流程 图 如 图 3.21 所 示 。 

例 3.11 本 例 将 使 用 for 循环 输出 数组 变量 。 如 果 变 量 的 数组 下 标 为 偶数 ， 则 只 输出 一 个 空 行 ， 如 果 是 
奇数 ， 才 继续 输出 。 在 最 里 面 的 循环 中 ， 判 断 当 前 数组 下 标 是 否 等 于 $i， 如 果 不 相等 ， 输 出 数组 变量 ， 否 则 
跳 到 最 外 重 循环 。〔 实 例 位置 : 光盘 \TM\Instance\03\3.11) 


代码 如 下 : 
<?php 
$arr = array("PHP 程序 开发 实战 宝典 ","PHP 求职 宝典 ","PHP 典型 模块 ","PHP 项 目 开发 全 程 实录 ","PHP 开发 
实战 1200 例 ","PHP 网 络 编程 自学 手册 "); // 声 明 一 个 数组 变量 $arr 
for($i = 0; $i < 6; $i++){ // 使 用 for 循环 
echo "<br>"; 
if($i % 2 == OX{ // 如 果 $i 的 值 为 偶数 ， 则 跳出 本 次 循环 
continue; 
} 
for(;;{ // 无 限 循环 
for($j = 0; $ < count($arr); Sj++X{ // 再 次 使 用 for 循环 输出 数组 变量 
if($j == $iX{ // 如 果 当 前 输出 的 数组 下 标 等 于 $i 
continue 3; // 跳 出 最 外 重 循环 
}else{ 
echo Sarr[$j].™\t | "; /| 输出 表达 式 
} 
} 
外 
echo "不 会 看 到 "; 
?> 


运行 结果 如 图 3.22 所 示 。 
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3.21 continue 跳 转 语句 的 流程 图 3.22 ”使 用 continue 语句 控制 流程 的 应 用 
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3.5.1 执行 指定 次 数 的 循环 


例 3.12 利用 break 语句 实现 执行 指定 次 数 的 循环 。《〈 实 例 位 置 ; 光盘 \TM\Instance\03\3.12) 
代码 如 下 : 
<table width="761" border="0" align="center "cellpadding="0" cellspacing="0" bordercolor="#FEFEFE" bgcolor= 
#FFFFFF"> 
<form action="" method="post" name="form1" id="form1"> 
<tr> 
<td width="761" align="center” bgcolor="#F9F8EF"><table width="749" border="0" align="center" 
cellpadding="0" cellspacing="0" style="BORDER-COLLAPSE: collapse"> 
<tr> 
<td width="749" height="57" background="images/a_03.jpg">&nbsp;&nbsp;</td> 
</tr> 
<tr> 
<td height="36" colspan="3" align="left" background="images/a_05.jpg" bgcolor="#F9F8EF" 
Scope="col"> 姓 名 : 
<input name="user_name" id="user_name" value="” 匿名 " maxlength="64" type="text" /> 
<span 
style="COLOR: #ff0000">*</span></td> 
</tr> 
<tr> 
<td height="36" colspan="3" align="left' background="images/a_05.jpg" bgcolor="#F9F8EF"> 标 题 : 
<input maxlength="64" size="30" name="title" type="text"/> 
<span style="COLOR: #f0000">*</span></td> 
</tr> 
<tr> 
<td height="126" colspan="3" align="left" backgroun 
bgcolor="#F9F8EF">&nbsp;&nbsp; &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 内 &nbsp;&nbsp;: 
<textarea name="content" cols="60" rows="8" id="content" style="background:url 
(Jimages/ mrbccd.gif)"></textarea> 
<span style="COLOR: #ff0000">*</span></td> 
</tr> 
<tr> 
<td height="40" colspan="3" align="center" background="images/a_05.jpg" bgcolor="#F9F8EF"> 
<input name="submit" type="submit" class="btn1" id="submit" value=" 签 写 留言 "/> 
<input name="reset" type="reset" class="btn1" value=" 清 除 留言 " /> 
</td> 
</tr> 
</table> 
</td> 
</tr> 
<tr> 
<td width="761" align="center" bgcolor="#F9F8EF"> 
<table width="749" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 


"images/a_05.jpg" 
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<td width="703" height="31" align="center" background="images/a_07.jpg">&nbsp;</td> 
</tr> 
</table></td> 


</form> 

</table> 

当 用 户 发 表 留 言 信 息 后 ， 提 交 留 言 信息 时 ， 通 过 正则 表达 式 preg_matchO 函 数 将 留言 信息 与 存储 在 数组 
中 的 敏感 词 进行 对 比 ， 检 验 用 户 提交 的 留言 信息 中 是 否 含有 敏感 词 。 如 果 留 言 信息 中 含有 敏感 词 ， 那 么 将 
弹出 提示 信息 ， 否 则 留言 信息 发 布 成 功 。 实 现 敏感 词 过 滤 的 完整 代码 如 下 : 

<?php 

if($_POST[sub1]X{ 
for($a = 0;$a < 10000000000000;$a++){ 
if($a >= $_POST[textl){ 
break; 


}else{ 
echo "已 经 输出 "($a+1)." 次 循环 <br>"; 
} 
日 
} 


?> 
运行 结果 如 图 3.23 所 示 。 


3.5.2 ” 跳 过 数据 输出 中 指定 的 记录 


例 3.13 ”应 用 continue 语句 跳 过 指定 的 循环 , 输出 指定 循环 以 外 的 数据 。( 实 例 位 置 : 光盘 \TM\Instance\ 
03\3.13) 

代码 如 下 : 
<?php 

$array = array("PHP 典型 模块 ","PHP 开发 实战 宝典 ","JAVA 开发 实战 宝典 ","PHP 网 络 编程 自学 手册 "); 
?> 
<form action="" method="post"> 

<textarea name="te" cols="20" rows="6">0、<?php echo $array[0]?> 1、<?php echo $array[1]?> 
2、<?php echo $array[2]?> 3、<?php echo $array[3]?> 

</textarea><br> 

<input type="text" name="text" value=" 输 入 要 跳 过 记录 的 编号 "size="20" onFocus="this.value=""> 

<input type="submit" name="sub1" value=" 跳 过 "> 
</form> 
<?php 

if($_POST[sub1]X{ 

for($a = 0;$a < count($array):$a++){ 
if($a == $_POST[text]}{ 


continue; 
}else{ 
echo' &' 
echo $array[$a]; 
echo '》 
} 
} 
} 
?> 
运行 结果 如 图 3.24 所 示 。 
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图 3.23 执行 指定 次 数 的 循环 图 3.24 ” 跳 过 数据 输出 中 指定 的 记录 


3.5.3 ”控制 页 面 中 数据 的 输出 数量 


例 3.14 运用 break 语句 跳出 指定 的 几 重 循 环 。〈 实 例 位 置 ， 光盘 \TM\Instance\03\3.14) 


代码 如 下 : 
<?php 
for($a = 0;$a < 2,$a++\{ 
for($b = 0;$b < 2;$b++){ 
for($c = 0;$c < 4;$c++) 
Srand = rand(1,4); 
echo "<img src='image/$rand.gif>"; 


} 


} 
echo "<hr style='color:blue;'>"; 
for($a = 0;$a < 2;$a++){ 
for($b = 0;$b < 2;$b++){ 
for($c = 0;$c < 4;$c++j{ 
Srand = rand(1,4); 
if($rand != 2X{ 
echo "<img src='image/$rand.gif >"; 


break 3; 


. 
?> 
运行 本 例 ， 单 击 栏目 导航 条 中 的 超 链接 ， 将 在 主 显示 区 中 [ERARRERRRESREE 
输出 对 应 的 项 目 内 容 ， 如 图 3.25 所 示 。 ET EY (E77 
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例 3.15 通过 使 用 continue 语 句 实现 动态 改变 页 面 中 单元 
格 的 背景 颜色 。 (实例 位 置 : 光盘 \TMNVInstance\03\3.15) 


代码 如 下 : 
<?php 


图 3.25 控制 页 面 中 数据 的 输出 数量 


for($b = 0;$b < 10;$b++){ 
if($b <= 5){ 
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Srand.= dechex(rand(0,15)); 


continue; 


中 
yw 
<table height="55" bgcolor="#<?php echo $rand;?>"> 
<tr><td width="156"></td> 
</tr></table> 


运行 结果 如 图 3.26 所 示 。 
3.5.5 ”使 用 for 循环 动态 创建 表格 


例 3.16 ”通过 使 用 for 循环 动态 创建 表格 。〈 实 例 位 置 : 光盘 \TMNInstance\03\3.16) 


代码 如 下 : 
<?php 
for($a = 0;$a < 5:$a++){ 
echo "<table border='1'bordercolor=#000099' bordercolorlight='#000099' cellspacing="0'>"; 
echo "<tr>"; 
for($b = 0;$b <7;$b++}{ 
echo "<td>"; 
echo $b; 
echo "</td>"; 
} 
echo "</tr>"; 
echo "</table>"; 


?> : 
运行 结果 如 图 3.27 所 示 。 


- 


ET 


图 3.26 动态 改变 页 面 中 单元 格 的 背景 颜色 图 3.27 表格 的 动态 创建 


3.6 小 结 


本 章 主要 讲述 的 是 流程 控制 语句 的 知识 ， 并 且 对 算法 和 程序 的 控制 结构 进行 了 介绍 。 重 点 掌握 3 种 流 
程控 制 语句 一 一 条 件 控制 语句 、 循 环 控制 语句 和 跳 转 控制 语句 。 在 掌握 流程 控制 语句 的 基础 上 ， 对 程序 的 算 
法 和 控制 结构 应 该 有 所 了 解 。 读 者 通过 对 本 章 的 学 习 能 够 从 宏观 的 角度 去 认识 PHP 语言 ， 从 整体 上 形成 一 
个 开发 的 思路 ， 逐 渐 形 成 一 种 属于 自己 的 编程 思想 和 编程 方法 。 
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3.7 学习 成 果 检 验 


1. 应 用 switch…case 分 支 控制 语句 输出 本 年 度 当 前 月 份 的 活动 安排 。〈 实 例 位 置 ， 光盘 \TM\Instance\ 
03\3.17) 

2. 在 开发 动态 网 站 时 应 用 do…while 循环 语句 输出 企业 公告 信息 。 (实例 位 置 ， 光盘 \TM\Instance\ 
03\3.18) 

3. 使 用 循环 语句 ， 输 出 任意 一 个 二 维 数组 。 (实例 位 置 : 光盘 \TM\Instance\03\3.19) 

4. 使 用 循环 控制 语句 ， 输 出 杨辉 三 角 。〔 实 例 位 置 ， 光 盘 \TM\Instance\03\3.20) 
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字符 串 操 作 与 正则 表达 式 


( 册 视频 讲解 : 92 分钟 ) 


在 Web 编程 中 , 字符 串 总 是 会 被 大 量 地 生成 和 处 理 。 正 确 地 使 用 
和 处 理 字符 串 ， 对 于 PHP 程序 员 来 说 越 来 越 重要 。 并 且 在 新 技术 层 出 
不 穷 的 今天 ， 让 人 难忘 的 、 能 称 得 上 是 伟大 技术 的 容 富 无 几 ， 但 其 中 
一 定 会 有 正则 表达 式 ， 然 而 ,， 最 容易 被 人 和 名 略 和 遗忘 的 也 是 正则 表达 
式 。 本 章 从 最 简单 的 字符 串 定义 开始 讲解 正则 表达 式 的 知识 ， 和 希望 广 
大 读者 通过 本 章 的 学 习 ， 了 解 和 党 扣 PHP 字符 串 和 正则 表达 式 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

Wm 掌握 单 引号 和 双 引 号 的 区 别 

Mm 掌握 字符 吝 的 连接 方法 

Wm 熟悉 去 除 字符 吝 中 的 空格 的 方法 

Mi 掌握 查找 和 侍 换 字 符 吝 技术 

MH 了 解 正 则 表达 式 的 发 展 及 相关 概念 

mm 了 解 PHP 中 的 POSIX 国 数 

MH 了 解 PHP 中 的 PCRE 国 数 

”掌握 正则 表达 式 的 应 用 
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4.1 了 解 字符 串 


字符 串 是 由 零 个 或 多 个 字符 组 成 的 有 限 序 列 。 字 符 包含 以 下 几 种 类 型 : 

数字 类 型 ， 如 1、2、3 等 。 

字母 类 型 ， 如 a、b、c、d 等 。 

特殊 字符 ， 如 $、%、#、^、 及 等 。 

不 可 见 字符 ， 如 \t (Tab 字符 ) 、 (换行 符 ) 、\r( 回 车 符 ) 等 。 

其 中 ， 不 可 见 字符 是 比较 特殊 的 一 组 字符 ， 是 用 来 控制 字符 串 格式 化 输出 的 ， 在 浏览 器 上 不 可 见 ， 只 
能 看 到 字符 串 输出 的 结果 。 

例如 : 

<?phl 


echo "PHP \t SQL Server n MySQL \r Java"; “”// 输 出 字符 串 
2 


结果 为 : PHP SQL Server 
MySQL 
Java。 


et 
看 源 文件 ”命令 来 查看 执行 不 可 见 字符 囊 的 实际 输出 结果 


4.2 单 引号 与 双 引 号 


敬 i 视频 讲解 :光盘 \TM\Video\ 第 4 章 \ 单 引号 与 双 引 号 .exe 

字符 串通 常 以 串 的 整体 作为 操作 对 象 ， 一 般 用 双 引 号 或 单 引 号 标识 一 个 字符 串 。 单 引号 串 和 双 引 号 串 
在 使 用 上 有 一 定 区 别 。 

下 面 分 别 使 用 单 引号 和 双 引 号 来 定义 一 个 字符 串 。 


<?php 

$str = 6; 

echo "str is $str"; /| 输出 结果 : str is 6 

echo 'str is $str'; /| 输出 结果 : str is $str 

echo "str is $str\n"; // 输 出 结果 : str is 6 〈 同 时 换行 ) 
echo 'str is $str\in'; /| 输出 结果 : str is $strin 

了 


从 以 上 代码 中 可 以 看 出 ， 双 引号 中 的 内 容 是 经 过 PHP 的 语法 分 析 器 解析 过 的 ， 任 何 变量 在 双 引 号 中 都 
会 被 转换 为 它 的 值 进行 输出 显示 ; 而 单 引 号 的 内 容 是 “所 见 即 所 得 ”的 ， 无 论 有 无 变量 ， 都 被 当 作 普 通 字 
符 串 进行 原样 输出 。 

然而 ， 对 于 定义 的 普通 字符 串 却 显示 不 出 两 者 之 间 的 区 别 。 例 如 : 


<?php 

$str1 = "| Like eat"; /应 用 双 引 号 定义 一 个 字符 串 
$str2 = "| Like eat'; /应 用 单 引 号 定义 一 个 字符 串 
echo $str1; /输出 双 引 号 中 的 字符 串 
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echo $str2; /输出 单 引号 中 的 字符 串 


?> 
结果 为 : ILike eat 
I Like eat。 


， 


二 


a 
音 引 号 囊 中 的 内 容 被 作为 普通 字符 进行 处 理 。 


43 定 界 符 


定 界 符 (<<<) 是 从 PHP 4.0 开始 支持 的 。 定 界 符 用 于 定义 格式 化 的 大 文本 ， 格 式 化 是 指 文本 中 的 格式 
将 被 保留 ， 所 以 文本 中 不 需要 使 用 转 义 字符 。 在 使 用 时 后 接 一 个 标识 符 ， 然 后 是 格式 化 文本 〈 即 字符 串 ) ， 
最 后 是 同样 的 标识 符 结束 字符 串 。 


str; 

其 中 ,符号 “<<<” 是 关键 字 ， 必 须 使 用 ，str 为 用 户 自 定义 的 标识 符 ， 用 于 定义 文本 的 起 始 标识 符 和 结 
束 标识 符 ， 前 后 的 标识 符 名 称 必 须 完全 相同 。 

结束 标识 符 必须 从 行 的 第 一 列 开 始 。 而 且 也 必须 遵循 PHP 中 其 他 标签 的 命名 规则 : 只 能 包含 字母 、 数 
字 、 下 划 线 ， 而 且 必须 以 下 划 线 或 非 数 字 字符 开始 。 

例 4.1 应 用 定 界 符 输出 变量 中 的 值 ， 可 以 看 到 ， 它 和 双 引 号 没什么 区 别 ， 包 含 的 变量 也 被 蔡 换 成 实际 
数值 。〈 实 例 位 置 : 光盘 \TMNInstance\04\4.1) 

代码 如 下 : 

<?php 

$str=" 获 请 大 家 "; 

echo <<<strmark 

<font color="#FF0099"> $str 关注 明日 图 书 网 : www.mingribook.com </font> 


strmark; 
?> 


在 上 面 的 代码 中 ， 值 得 注意 的 是 ， 在 定 界 符 内 不 允许 添加 注释 ， 否 则 程序 将 运行 出 错 。 结 束 标识 符 所 
在 的 行 不 能 包含 任何 其 他 字符 ， 而 且 不 能 被 缩 进 ， 在 标识 符 分 号 前 后 不 能 有 任何 空白 字符 或 制 表 符 。 如 果 
破坏 了 这 条 规则 ， 则 程序 不 会 被 视 为 结束 标识 符 ，PHP 将 继续 寻找 下 去 。 如 果 在 这 种 情况 下 找 不 到 合适 的 
结束 标识 符 ， 将 会 导致 一 个 在 脚本 最 后 一 行 出 现 的 语法 错误 。 
运行 结果 如 图 4.1 所 示 。 


EF 技巧 定 界 符 中 的 字符 串 支 持 单 引号 、 双 引号 ， 无 须 转 义 ， 并 支持 字符 变量 蔡 换 。 


例 4.2 应 用 定 界 符 输出 一 个 用 户 登录 的 表单 。〔 实 例 位置 ， 光盘 \TM\Instance\04\4.2) 

代码 如 下 : 

<?php 

$str=<<<user 

<IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/ 
xhtml1- transitional.dtd"> 


@_ 
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<html xmlns="http:/www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title> 用 户 登录 </title> 
<link rel="stylesheet" type="text/css" href="style.css" /> 
<body> 
<table width="250" height="100" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td bgcolor="#666666"><table width="250" height="100" border="0" cellpadding="0" cellspacing="1"> 
<form name="form1" method="post" action="index.php"> 
<tr> 
<td height="25" colspan="2" bgcolor="#DDAACC"><div align="center" > 用 户 登录 </div></td> 
</tr> 
<tr> 
<td width="60" height="25" bgcolor="#FFFFFF"><div align="right"> 用 户 名 : </div></td> 
<td width="187" bgcolor="#FFFFFF">&nbsp;<input type="text" name="ip" size="18" 
class="inputcss"></td> 
</tr> 
<tr> 
<td height="25" bgcolor="#FFFFFF"><div align="right"> 密 码 : </div></td> 
<td height="25" bgcolor="#FFFFFF">&nbsp;<input type="text” name="ipNum" size="18" 
class="inputcss"> </td> 
</tr> 
<tr> 
<td height="25" colspan="2" bgcolor="#FFFFFF"><div align="center"><input name="submit" 
type="submit" class="buttoncss" value=" 登 录 " /> 
</div></td> 
</tr> 
</form> 
</table></td> 
</tr> 
</table> 


人 注意 在 上 面 的 代码 中 ， 结 束 标识 符 user 必须 单独 另 起 一 行 ， 并 且 不 允许 有 空白 字符 。 如 果 在 标识 
符 前 后 有 其 他 符号 、 字 符 或 注释 ， 则 会 发 生 错 误 。 


运行 结果 如 图 4.2 所 示 。 
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图 4.1 使 用 定 界 符 定义 字符 串 图 4.2 使 用 定 界 符 输出 用 户 登 录 表 单 
区 
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4.4 连接 字符 串 


铭 1 视频 讲解 : 光盘 \TM\Video\ 第 4 章 \ 连 接 字符 囊 .exe 

字符 串 可 以 用 “.” 点 ) 字符 串 连 接 符 连接 ， 它 可 以 把 两 个 或 两 个 以 上 的 字符 串 连 接 成 一 个 新 的 字符 
串 。 注 意 这 里 不 能 用 “+” (加 ) 运算 符 。 

字符 串 连接 有 两 种 形式 ， 第 一 种 是 连接 运算 符 “.”; 第 二 种 是 连接 赋值 运算 符 “.=”。 

连接 运算 符 “.” 返 回 其 左右 参数 连接 后 的 字符 串 。 例 如 : 

<?php 

$name=" 阴 日 图 书 网 :"; 

$url = $name."www.mrbccd.com"; 

echo $url; 

?> 


结果 为 : 明日 图 书 网 ，www.mrbccd.com。 

连接 赋值 运算 符 “.=” 将 右边 参数 附加 到 左边 的 参数 后 连接 成 一 个 新 的 字符 串 。 例 如 : 
<?php 

$url=" 了 明日 图 书 网 :"; 

$url .= "www.mrbccd.com "; 

echo S$url; 

?> 


结果 为 : 明日 图 书 网 ，www.mrbccd.com。 
4.5 转 义 、 还 原 字符 串 


在 PHP 编程 的 过 程 中 ， 经 常会 遇 到 这 样 的 问题 ， 将 数据 插入 到 数据 库 时 可 能 引起 一 些 问题 ， 出 现 错误 
或 者 乱码 等 ， 因 为 数据 库 将 传 入 的 数据 的 字符 解释 成 了 控制 符 。 针 对 这 种 问题 ， 就 需要 使 用 一 种 标记 或 者 
是 转 义 这 些 特殊 的 字符 。 

因此 ， 在 PHP 语言 中 提供 了 专门 处 理 这 些 问 题 的 技术 ， 转 义 和 还 原 字符 串 。 方 法 有 两 种 : 一 种 是 手动 
转 义 、 还 原 字 符 串 数据 ， 另 一 种 是 自动 转 义 、 还 原 字符 串 数据 。 下 面 分 别 对 这 两 种 方法 进行 详细 讲解 。 


4.5.1 手动 转 义 、 还 原 字 符 串 


字符 串 可 以 用 单 引号 () 、 双 引号 ("") 、 定 界 符 〈<<<) 3 种 方式 定义 。 而 指定 一 个 简单 字符 串 的 最 
简单 的 方法 是 用 单 引 号 〈") 括 起 来 。 当 使 用 字符 串 时 ， 很 可 能 在 该 串 中 存在 这 几 种 符号 与 PHP 脚本 混淆 的 
字符 ， 因 此 必须 要 做 转 义 语句 ， 即 在 它 的 前 面 使 用 转 义 符号 〈\) 。 

“\” 是 一 个 转 义 符 ， 紧 跟 在 “\” 后 面 的 第 一 个 字符 将 变 为 没有 意义 或 特殊 意义 。 例 如 ，“'” 是 字符 串 
的 定 界 符 ， 写 成 “\ ”这样 时 就 使 它 失 去 了 定 界 符 的 意义 ， 变 为 普通 的 单 引号 “'”。 读 者 可 以 通过 “echo N";” 
输出 一 个 单 引号 “'”， 同 时 转 义 字符 “\” 不 会 显示 。 


由 如 果 要 在 字符 囊 中 表示 单 引号 ， 则 需要 用 反 斜 村 “W 进行 转 义 。 例 如， 要 表示 字符 事 “Tm， 
则 需要 写成 “Tvm”。 
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例 4.3 使 用 转 义 字符 〈\) 对 字符 串 进 行 转 义 。 《实例 位 置 ， 光盘 \TMNInstance\04\4.3) 
代码 如 下 : 
<?php 
echo ' select * from book where bookname = \'PHP 开发 实战 宝典 \ '; 
> 


结果 为 : select * from book where bookname = 'PHP 开发 实战 宝典 '。 
区 5 
用 自动 转 义 函数 实现 字符 串 的 转 义 。 


4.5.2 自动 转 义 、 还 原 字符 串 


自动 转 义 、 还 原 字符 串 数据 可 以 应 用 PHP 提供 的 addslashes0 函 数 和 stripslashes0 函 数 实现 。 

addslashes( 〇 函数 

addslashes() 函 数 用 来 给 字符 串 str 加 入 反 和 斜 本 〈\) ， 对 指定 字符 串 中 的 字符 进行 转 义 ， 该 函数 可 以 转 义 
的 字符 包括 单 引 号 () 、 双 引号 (") 、 反 和 斜 杠 (\) 、NULL 字符 〈0) 。 该 函数 比较 常用 的 地 方 就 是 在 生 
成 SQL 语句 时 ， 对 SQL 语句 中 的 部 分 字符 进行 转 义 。 

语法 : 

string addslashes ( string str) 

参数 str 为 将 要 被 操作 的 字符 串 。 

stripslashes() 函 数 

stripslashes() 函 数 用 来 将 应 用 addslashes0 函 数 转 义 后 的 字符 串 str 返回 原样 。 

语法 : 

string stripslashes(string str); 

参数 str 为 将 要 被 操作 的 字符 串 。 

例 4.4 使 用 自动 转 义 字符 addslashes0 函 数 对 字符 串 进 行 转 义 ， 然 后 应 用 stripslashes0 函 数 进行 还 原 。 
(实例 位 置 ， 光盘 \TMNInstance\04\4.4) 


代码 如 下 : 
<?php 
$str = "select * from book where bookname = 'PHP 开发 实战 宝典 "; 
$a = addslashes($str); // 对 字符 串 中 的 特殊 字符 进行 转 义 
echo $a."<br>"; // 输 出 转 义 后 的 字符 
$b = stripslashes($a); // 对 转 义 后 的 字符 进行 还 原 
echo $b."<br>"; // 将 字符 原 义 输出 
?> 
运行 结果 如 图 4.3 所 示 。 


罗拉 万 所 有 数据 在 插入 数据 库 前 ， 有 必要 应 用 addslashes0 函 数 进行 字符 串 转 义 ， 以 免 特殊 字符 未 经 
转 义 在 插入 数据 库 时 出 现 错误 。 另 外 ， 对 于 应 用 addslashes0) 函 数 实现 的 自动 转 义 字符 串 可 以 应 用 
stripslashesO 函 数 进行 还 原 ， 但 数据 在 插入 数据 库 前 必须 再 次 进行 转 义 。 


以 上 两 个 函数 实现 了 对 指定 字符 串 进 行 自动 转 义 和 还 原 。 除 了 上 面 介绍 的 方法 外 ， 还 可 以 对 要 转 义 、 
还 原 的 字符 串 进行 一 定 范围 的 限制 , PHP 通过 应 用 addcslashes0 函 数 和 stripcslashes0 函 数 实现 对 指定 范围 内 


图 
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的 字符 串 进行 自动 转 义 、 还 原 。 

addcslashes0 〇 函数 

实现 对 指定 字符 串 中 的 字符 进行 转 义 ， 即 在 指定 的 字符 charlist 前 加 上 反 斜 枉 〈\) 。 通 过 该 函数 可 以 将 
要 添加 到 数据 库 中 的 字符 串 进行 转 义 ， 从 而 避免 出 现 乱 码 等 问题 。 

语法 : 

string addcslashes ( string str, string charlist) 

参数 str 为 将 要 被 操作 的 字符 串 ， 参数 charlist 指定 在 字符 串 中 哪些 字符 前 加 上 反 斜 枉 〈\) ， 如 果 参 数 
charlist 中 包含 有 “\n”、“\r” 等 字符 ， 将 以 C 语言 风格 转换 ， 而 其 他 非 字母 数字 且 ASCII 码 低 于 32 或 高 
于 126 的 字符 均 转换 成 以 八进制 表示 。 


Py 
ea 


stripcslashes() 函 数 

stripeslashes0 〇 函数 用 来 将 应 用 addcslashes0 函 数 转 义 的 字符 串 str 还 原 。 

语法 : 

string stripcslashes ( string str) 

参数 str 为 将 要 被 操作 的 字符 串 。 

例 4.5 应 用 addcslashesO 函 数 对 字符 串 “ 明 日 科技 ”进行 转 义 ， 应 用 stripeslashes0 函 数 对 转 义 的 字符 
串 进 行 还 原 。《〈 实 例 位 置 : 光盘 \TM\Instance\04\4.5) 


代码 如 下 : 

<?php 

$a=" 阴 日 科技 "; // 对 指定 范围 内 的 字符 进行 转 义 
$b=addcslashes($a," 明 日 科技 "); // 转 义 指定 的 字符 串 

echo " 转 义 字符 串 : ".$b; /输出 转 义 后 的 字符 串 

echo "<br>"; // 执 行 换行 
$c=stripcslashes($b); // 对 转 义 的 字符 串 进行 还 原 
echo "还 原 字符 串 : ".$c; // 输 出 还 原 后 的 转 义 字符 串 

?> 


运行 结果 如 图 4.4 所 示 。 


Er 


窜 6 二 BE -| 卉 包 . XX 尼 loce- 全- 日 ， 


select + from bock where booknane = 下 a 
| select + 三 am bock where bcoknanc = “PtP 开发 实战 宇 | 
| ET ee 


图 4.3 ”自动 转 义 和 还 原 字符 串 图 4.4 对 指定 范围 的 字符 串 进行 转 义 和 还 原 


-和 


【> 技 瑟 缓存 文件 中 ， 一 般 对 缓存 数据 的 值 采用 addcslashesO 函 数 进行 指定 范围 的 转 义 。 


4.6 获取 字符 串 长 度 


名 4 视频 讲解 :光盘 \TM\Video\ 第 4 章 \ 获 取 字 符 囊 长 度 .exe 
获取 字符 串 长 度 主要 通过 sttlen0 函 数 实现 ， 下 面 重点 讲解 strlen0 函 数 的 语法 及 其 应 用 。 


局 
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strlen0 函 数 主 要 用 于 获取 指定 字符 串 str 的 长 度 。 
语法 : 
int strlen(string str) 

例 4.6 应 用 stlen0 函 数 来 获取 指定 字符 串 的 长 度 。 (实例 位 置 : 光盘 \TM\Instance\04\4.6) 
代码 如 下 : 

<?php 


echo strlen(" 阴 日 科技 图 书 网 :www.mingribok.com"); /! 输 出 指定 字符 串 长 度 


?> 


结果 为 : 32。 


位 明 ， 


汉字 占 两 个 字符 ， 数 字 、 英 文 、 小 数 点 、 下 划 线 和 空格 各 占 一 个 字符 。 


strlen0 函 数 在 获取 字符 串 长 度 的 同时 ， 也 可 以 用 来 检测 字符 串 的 长 度 。 
例 4.7 应 用 strlen0 函 数 对 提交 的 用 户 密码 的 长 度 进行 检测 ， 如 果 其 长 度 小 于 4 位 ， 则 弹出 提示 信息 。 
《实例 位 置 ， 光盘 \TM\Instance\04\4.7) 
(1) 利用 开发 工具 (如 Dreamweaver) ， 新 建 一 个 PHP 动态 页 ， 存 储 为 index.php。 
(2) 添加 一 个 表单 ， 将 表单 的 action 属性 设置 为 index_ok.php。 
<form name="form1" method="post" action="index_ok.php"> 
<tr> 
<td height="33"></td> 
<td align="center"><span class="style1"> 用 户 名 </span>:</td> 
<td> 
<input name="user" type="text" id="user" size="15"> 
</td> 
<td align="center"> 
<input name="imageField" type="image" src="images/btn_dl.jpg" width="50" height="20" 
border="0"> 
</td> 
<td>&nbsp;</td> 
</tr> 
<tr> 
<td height="27">&nbsp;</td> 
<td align="center"><span class="style1"> 密 码 </span>:</td> 
<td> 
<input name="pwd" type="password" id="pass" size="15"> 
</td> 
<td colspan="2" align="left"><span class="STYLE3">*&nbsp; 密 码 长 度 不 能 少 于 4 位 </span></td> 
</tr> 
</form> 


(3) 应 用 HTML 标记 设计 页 面 ,添加 一 个 “用 户 名 ”文本 框 ， 命 名 为 user; 添加 一 个 “密码 ”文本 框 ， 
命名 为 pwd; 添加 一 个 图 像 域 ， 指 定 源 文件 位 置 为 images/btn_dljpg。 

(4) 再 新 建 一 个 PHP 动态 页 ， 存 储 为 index_ok.php。 通 过 POST 方法 (关于 POST 方法 将 在 后 面 章节 
中 进行 详细 讲解 ) 接收 用 户 输入 的 用 户 密码 的 值 。 应 用 strlen0 函 数 获 取 用 户 提交 密码 的 长 度 ， 应 用 寺 条 件 
控制 语句 判断 密码 长 度 是 否 小 于 4， 并 给 出 相应 的 提示 信息 。 代 码 如 下 : 


<?php 

if(strlen($_POST["pwd"])<4){ /检测 用 户 密码 的 长 度 小 于 4， 弹出 警告 信息 
echo "<script>alert(' 用 户 密码 的 长 度 不 得 少 于 4 位 ! 请 重新 输入 '); history.back();</script>"; 

} 

else{ /用 户 密码 大 于 等 于 4 位 ， 则 弹出 该 提示 信息 
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echo "用 户 信息 输入 合法 ! “"; 
日 


x 


(5) 在 下 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 运 行 结果 如 图 4.5 所 示 。 


企业 写 眼 管理 系统 


4.5 应 用 strlen0 函 数 检测 字符 串 的 长 度 
4.7 截取 字符 串 


名 m 视频 讲解 :光盘 TM\Video\ 第 4 章 \ 截 取 字符 囊 .exe 

在 PHP 中 ， 有 一 项 非常 重要 的 技术 ， 就 是 截取 指定 字符 串 中 指定 长 度 的 字符 。PHP 对 字符 串 截取 可 以 
通过 PHP 的 预定 义 函 数 substr0 实 现 。 本 节 重 点 介绍 字符 串 的 截取 技术 。 

substr0 函 数 从 字符 串 中 按照 指定 位 置 截 取 一 定 长 度 的 字符 。 通 过 该 函数 可 以 获取 某 个 固定 格式 字符 串 
中 的 一 部 分 ， 如 果 使 用 一 个 正 数 作为 子 串 起 点 来 调用 这 个 函数 ， 将 得 到 从 起 点 到 字符 串 结束 的 这 个 字符 串 ， 
如 果 使 用 一 个 负数 作为 子 串 起 点 来 调用 ， 将 得 到 一 个 原 字符 串 尾部 的 子 串 ， 字 符 个 数 等 于 给 定 负数 的 绝对 值 。 

语法 ， 

string substr ( string str, int start [, int length]) 

参数 说 明 如 表 4.1 所 示 。 


表 4.1 substr 函数 的 参数 说 明 


指定 字符 串 对 象 


指定 开始 截取 字符 串 的 位 置 ， 如 果 参 数 start 为 负数 ， 则 从 字符 串 的 末尾 开 
指定 截取 字符 的 个 数 ， 如 果 length 为 负数 ， 则 表示 截取 到 倒数 第 length 个 字符 


sa 


例 4.8 应 用 substr0 函 数 截取 字符 串 中 指定 长 度 的 字符 。《〈 实 例 位 置 : 光盘 \TMNInstance\04\4.8) 
代码 如 下 : 

<?php 

echo substr("www.mingribook.com",0); // 从 第 0 个 字符 开始 截取 

echo "<br>"; 

echo substr("www.Mingribook. ",3,8); /从 第 3 个 字符 开始 连续 截取 8 个 字符 

echo "<br>"; 


echo substr("www.Mingribook.com ",-3,3); /从 倒数 第 3 个 字符 开始 截取 3 个 字符 
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echo "<br>"; 
echo substr("www.Mingribook.com ",0,-2); // 从 第 一 个 字符 开始 截取 ， 截 取 到 倒数 第 2 个 字符 


?> 
结果 为 : www.mingribook.com 
mingrib 
om 
www.mingribook.co。 
例 4.9 下面 应 用 substr0 函 数 截取 超 长 文本 的 部 分 字符 串 ， 剩 余 的 部 分 用 “…” 代 蔡 。《〈 实 例 位置 : 光 
盘 \TMNVInstance\04\4.9) 


代码 如 下 : 
<?php 
$str=" 明 日 科技 有 限 公司 是 一 家 知名 企业 的 软件 公司 ， 公 司 主要 经 营 图 书 开发 词典 ， 各 种 图 书 。"; 
if(strlen($str)>40){ // 如 果 文本 的 字符 串 长 度 大 于 40 
echo substr($str,0,40)."..."; // 输 出 文本 的 前 40 个 字符 串 ， 然 后 输出 省 略 号 
} 
else{ // 如 果 文本 的 字符 串 长 度 小 于 40 
echo $str; /直接 输出 文本 
} 
?> 


结果 为 :明日 科技 有 限 公 司 是 一 家 知名 企业 的 软件 公司 .…。 
4.8 比较 字符 串 


在 PHP 中 ， 对 字符 串 之 间 进 行 比较 的 方法 有 很 多 种 : 第 一 种 应 用 strcmpO 和 streasecmp0 函 数 按照 字 节 
进行 比较 ， 第 二 种 应 用 strmatemp0 函 数 按照 自然 排序 法 进行 比较 ， 第 三 种 应 用 strmemp0 函 数 指定 从 源 字符 
串 的 位 置 开 始 比较 。 下 面 分 别 对 这 3 种 方法 进行 讲解 。 


4.8.1 按 字 节 比较 


按 字 节 进 行 字符 串 比 较 的 方法 有 两 种 ， 分 别 为 使 用 stremp0 函 数 和 strecasecmp0 函 数 进行 比较 。 这 两 种 
函数 的 区 别 是 strempO 函 数 区 分 字符 的 大 小 写 ， 而 streasecmp0 函 数 不 区 分 字符 的 大 小 写 。 这 两 个 函数 的 实 
现 方法 基本 相同 ， 这 里 只 介绍 stremp0 函 数 。 

stremp() 函 数 用 来 对 两 个 字符 串 进行 比较 。 

语法 : 

rs ( string str1, string str2)) 

参数 strl 和 参数 str2 指定 要 比较 的 两 个 字符 串 。 如 果 相 等 则 返回 0， 如 果 参 数 strl 大 于 参数 st2， 则 返 
回 1; 如 果 参 数 strl 小 于 参数 st2， 则 返回 -1。 


人 注意 该 函数 区 分 字母 大 小 写 。 


例 4.10 ”应 用 strcmpO0 和 strcasecmpO 函 数 分 别 对 两 个 字符 串 按 字 节 进行 比较 。 (实例 位 置 ， 光盘 \TM 
Instance\04\ 4.10) 


四 
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代码 如 下 : 

<?php 

$str1=" 阴 日 科技 人"; /定义 字符 串 常量 
Sstr2=" 了 明日 科技 人" /定义 字符 串 常量 
S$str3="mrsoft"; /定义 字符 串 常量 
$str4="MRSOFT"; // 定 义 字 符 串 常量 

echo strcmp($str1,$str2); // 这 两 个 字符 串 相等 

echo strcmp($str3,$str4); /注意 该 函数 区 分 大 小 写 
echo strcasecmp($str3,$str4); // 该 函数 不 区 分 字母 大 小 写 
?> 


结果 为 : 0 1 0。 


\ 


| stremp0 函 数 来 比较 在 
用 户 登 录 系 统 中 输入 的 用 户 和 密码 是 否 正确 。 如 果 在 验证 用 户 名 和 密码 时 不 使 用 此 函数 ， 那 么 输入 的 用 
户 名 和 密码 无 论 是 大 写 还 是 小 写 ， 只 要 正确 就 可 以 登录 。 应 用 了 strcmp0 函 数 之 后 就 避免 了 这 种 情况 ， 
即使 正确 ， 必 须 大 小 写 匹配 才 可 以 登录 ， 从 而 提高 了 网 站 的 安全 性 。 


4.8.2 按 自然 排序 法 比较 


在 PHP 中 ， 按 照 自然 排序 法 进行 字符 串 比 较 是 通过 strnatcmp0 函 数 来 实现 的 。 自 然 排序 法 比较 的 是 字 
符 串 中 的 数字 部 分 ， 将 字符 串 中 的 数字 按照 大 小 进行 比较 。 

strnatemp0 函 数 通过 自然 排序 法 比较 字符 串 。 

语法 : 

int strnatcmp ( string str1, string str2) 

如 果 字 符 串 相等 ， 则 返回 0， 如 果 参 数 strl 大 于 参数 st2， 则 返回 1; 如 果 参 数 strl 小 于 参数 st2， 则 
返回 -1。 本 函数 区 分 字母 大 小 写 。 


注意 从 拓 设 中 ， 2 比 10 小 。 在 计算 机 序列 当中 ，10 比 2 小 ， 因 为 “10” 中 的 第 一 个 数字 是 
232 


例 4.11 应 用 strmatemp0 函 数 按 自然 排序 法 进行 字符 串 的 比较 。( 实 例 位 置 : 光盘 \TMNInstance\04\4.11) 


代码 如 下 : 

<?php 

S$str1="str2.jpg"; /定义 字符 串 常量 
S$str2="str10.jpg"; /定义 字符 串 常量 
$str3="mrsoft1"; /定义 字符 串 常量 
$str4="MRSOFT2"; // 定 义 字符 串 常 量 

echo strcmp($str1,$str2); // 按 字 节 进行 比较 ， 返 回 1 
echo strcmp($str3,$str4); // 按 字 节 进行 比较 ， 返 回 1 


echo strnatcmp($str1,$str2); 
echo strnatcmp($str3,$str4); 


2 
结果 为 : 1 1 -1 1。 


CC 


// 按 自然 排序 法 进行 比较 ， 返 回 -1 
// 按 自然 排序 法 进行 比较 ， 返 回 1 
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a. 
A 培 明 按照 自然 排序 法 进行 比较 ， 还 可 以 使 用 另 一 个 与 stmatcmp0O 函 数 相 同 ， 但 是 不 区 分 大 小 写 的 
stmatcasecmp(O) 函 数 ， 它 的 作用 和 strmatcmp(O 函 数 相同 。 


4.8.3 ”指定 从 源 字 符 串 的 位 置 比较 


stmemp0 函 数 用 来 比较 字符 串 中 的 前 n 个 字符 。 

语法 : 

int strncmp(string str1,string str2,int len) 

如 果 字 符 串 相等 ， 则 返回 0， 如 果 参 数 strl 大 于 参数 st2， 则 返回 1， 如 果 参 数 strl 小 于 参数 st2， 则 
返回 -1。 该 函数 区 分 字母 大 小 写 。 参 数 说 明 如 表 4.2 所 示 。 


表 4.2 stmcmp() 函 数 的 参数 说 明 


参数 说 明 
strl 指定 参与 比较 的 第 一 个 字符 串 对 象 
str2 指定 参与 比较 的 第 二 个 字符 串 对 象 
len 必 选 参数 ， 用 来 指定 每 个 字符 串 中 参与 比较 字符 的 数量 


例 4.12 应 用 stmcmp0O 函 数 比较 字符 串 的 前 5 个 字符 是 否 与 源 字符 串 相等 。〔 实 例 位 置 ， 光 盘 \TM\ 
Instance\04\4.12) 


代码 如 下 : 

<?php 

S$str1="| like life // 定 义 字符 串 常量 
$str2="i am a teacher!"; // 定 义 字符 串 常量 
echo strncmp($str1,$str2,5); /比较 前 6 个 字符 
Ee 

结果 为 : -1。 


从 上 面 的 代码 中 可 以 看 出 , 由 于 变量 $str2 中 的 字符 串 的 首 字母 为 小 写 , 与 变量 $strl 中 的 字符 串 不 匹配 ， 
因此 比较 后 的 字符 串 返回 值 为 -1。 


4.9 检索 字符 串 


在 PHP 中 ， 提 供 了 很 多 用 于 字符 串 查找 的 函数 ，PHP 也 可 以 像 Word 那样 实现 对 字符 串 的 查找 功能 。 
下 面 讲解 常用 的 字符 串 检索 技术 。 


4.9.1 使 用 strstr() 函 数 检索 指定 的 关键 字 
获取 一 个 指定 字符 串 在 另 一 个 字符 串 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字符 串 。 如 果 执 行 成 功 ， 则 返 
回 获取 的 子 字 符 串 (存在 相 匹 配 的 字符 ); 如 果 没 有 找到 相 匹配 的 字符 ， 则 返回 false。 


语法 : 
string strstr ( string haystack, string needle) 


@ 
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参数 说 明 如 表 4.3 所 示 。 
表 4.3 strstr() 函 数 的 参数 说 明 


参数 说 明 
haystack 必 选 参数 ， 用 来 指定 从 哪个 字符 串 中 进行 搜索 


必 选 参数 ， 用 来 指定 搜索 的 对 象 ， 如 果 该 参数 是 一 个 数值 ， 那 么 将 搜索 与 这 个 数值 的 ASCII 值 相 
匹配 的 字符 


needle 


《So 广 训 本 函数 区 分 字母 的 大 小 写 。 


例 4.13 应 用 strstr0 函 数 获取 指定 字符 串 在 字符 串 中 首次 出 现 的 位 置 后 的 所 有 字符 。 【实例 位 置 ; 光 
盘 \TMNInstance\04\4.13) 


代码 如 下 : 
<?php 
echo strstr(" 明 日 科技 图 书 网 "," 图 "); // 输 出 查询 的 字符 串 
echo "<br>"; // 执 行 换行 
echo strstr("http://www.mingribook.com","m"); /| 输出 查询 的 字符 串 
echo "<br>"; // 执 行 换行 
echo strstr("0431-84978981","8"); // 输 出 查询 的 字符 串 
?> 
结果 为 : 图 书 网 

mingribook.com 

84978981。 


通过 上 面 的 代码 可 以 看 出 ， 应 用 strstr0 函 数 自 定义 检索 字符 串 非常 方便 ， 另 外 ，strrchr0 函 数 与 其 正好 
相反 ， 该 函数 是 从 字符 串 后 序 的 位 置 开始 检索 子 串 的 。 


4.9.2 ”应 用 substr_count() 函 数 检索 子 串 出 现 的 次 数 


在 PHP 中 ， 有 一 种 技术 可 以 获取 字符 串 中 字符 和 单词 数量 ， 通 过 该 技术 可 以 查看 到 指定 字符 或 者 单词 
在 字符 串 中 出 现 的 次 数 ， 而 且 还 可 以 应 用 到 论坛 、 博 客 或 者 聊天 室 的 信息 发 布 模块 中 ， 判 断 提交 的 信息 中 
是 否 含有 非法 关键 字 。 

substr_count0 函 数 获取 指定 字符 在 字符 串 中 出 现 的 次 数 。 

语法 : 

int substr_count(string haystack,string needle) 

参数 haystack 是 指定 的 字符 串 ， 参 数 needle 为 指定 的 字符 。 

例 4.14 下 面 应 用 substr_countO 函 数 获取 子 串 在 字符 串 中 出 现 的 次 数 。( 实 例 位 置 : 光盘 \TM\Instance\ 
04\4.14) 


代码 如 下 : 

<?php 

$str=" 明 日 科技 图 书 网 图 书 "; 

echo substr_count($str," 书 "); // 输 出 查询 的 字符 串 
?> 

结果 为 : 2。 


从 表面 上 看 ， 该 函数 的 功能 就 是 获取 指定 字符 在 字符 串 中 出 现 的 次 数 ， 输 出 的 两 个 数字 ， 但 在 实际 的 
运用 中 ， 只 要 对 输出 的 数字 加 以 判断 后 ， 就 能 够 实现 不 同 功能 。 
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| 区 技 站 于 下 贞观 的 炎 喜 一 般 党 用 于 接 索 引 尝 中， 针对 于 事 在 字符 囊 中 出 现 的 大 堵 进 行 纯 计 ， 便于 
用 户 第 一 时 间 掌 握 子囊 在 字符 事 中 出 现 的 次 数 。 


4.10 替换 字符 串 


通过 字符 串 的 蔡 换 技 术 可 以 实现 对 指定 字符 串 中 的 指定 字符 进行 蔡 换 。 字 符 串 的 蔡 换 技 术 可 以 通过 以 
下 两 个 函数 实现 : str_ireplace0 和 substr_replace0 函 数 。 

1. str_irereplace() 函 数 

使 用 新 的 子 字 符 串 蔡 换 原始 字符 串 中 被 指定 要 蔡 换 的 字符 串 。 

语法 : 

mixed str_ireplace ( mixed search, mixed replace, mixed subject [, int &count] ) 

将 所 有 在 参数 subject 中 出 现 的 search 用 参数 replace 蔡 换 ， 参 数 &count 表示 蔡 换 字符 串 执行 的 次 数 。 
str_replace() 函 数 的 参数 说 明 如 表 4.4 所 示 。 

表 4.4 str_replace() 函 数 的 参数 说 明 


必 选 参数 ， 指 定 需要 查找 的 字符 串 


必 选 参数 ， 指 定 替换 的 值 
必 选 参数 ， 指 定 查找 的 范围 
可 选 参 数 ， 获 取 执行 蔡 换 的 数量 


Ce 
是 字符 串 还 是 数组 .如果 原始 字符 串 subject 是 一 个 数组 ， 则 该 函数 会 依次 用 replace 替换 subject 数组 中 
每 个 元 素 中 的 search 子 字符 串 ， 同 时 该 函数 的 返回 值 为 一 个 数组 。 


例 4.15 将 文本 中 的 指定 字符 串 “ 你 好 ” 蔡 换 为 “明日 科技 ”， 并 且 输 出 蔡 换 后 的 结果 。《〈 实 例 位 置 ; 


光盘 \TM\Instance\04\4.15) 
实例 代码 如 下 : 
<?php 
$str1 = "你 好 ”; // 定 义 字符 串 常量 
$str2 = "明日 科技 ”; /定义 字符 串 常量 
$str =“ 你 好 公司 是 一 家 以 编程 词典 技术 为 核心 的 高 科技 企业 ， 多 年 来 始终 致力 于 图 书 软件 的 开发 、 编 程 词典 的 销 
售 、 网 站 的 访问 日 益 增多 ”; /定义 字符 串 常量 
Echo str_ireplace($str1,$str2, $str); // 输 出 替换 后 的 字符 串 
2 
运行 结果 如 图 4.6 所 示 。 


全 = 注 意 该 函数 在 执行 替换 的 操作 时 不 区 分 大 小 写 , 如 果 需 要 对 大 小 写 加 以 区 分 , 可 以 使 用 str replace0 


辑 数 。 
OO 
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字符 串 昔 换 技术 最 常用 的 就 是 在 搜索 引擎 的 关键 字 处 理 中 ， 可 以 使 用 字符 串 蔡 换 技术 将 搜索 到 的 字符 
串 中 的 关键 字 蔡 换 颜 色 ， 如 查询 关键 字 描 红 功 能 ， 使 搜索 到 的 结果 更 便于 用 户 查看 。 


人 注意 查询 关键 字 描 红 是 指 将 查询 关键 字 以 特殊 的 颜色 、 字 号 或 字体 进行 标识 。 这 样 可 以 使 浏览 者 快 
束 检 索 到 所 需 的 关键 字 ， 方 便 浏览 者 从 搜索 结果 中 查找 所 需 内 容 。 查 询 关 键 字 描 红 适 用 于 模糊 查询 。 


例 4.16 使 用 str_ireplaceO 函 数 蔡 换 查询 关键 字 ， 当 显示 所 查询 的 相关 信息 时 ， 将 输出 的 关键 字 的 字体 
蔡 换 为 红色 。 【实例 位 置 ; 光盘 \TM\Instance\04\4.16) 

实例 代码 如 下 : 

<?php 
Sc = "吉林 省 明日 科技 有 限 公司 ， 是 一 家 知名 的 软件 公司 ， 明 日 主要 推出 编程 词典 ， 及 图 书 等 "; 
S$str=" 明 日 科技 "; 
echo str_ireplace($str,"<font color='#ff0000'>".$str."</font>",$c); 

?> 


二 本 加 当知 4.7 所 示 。 


Br wr 
Re /oes Mn 


- 国 - 一 帆 - mp ” 


阴间 村 楼 公司 是 一 家 凡生 检 枯 术 为 本 民 训 科技 企业 ， 多 年 直 外 念力 于 图 书 卓 
欣 休 的 开发 ， 居 各 的 滑雪 ， 同 站 的 访问 电 闪 地 公司 ， 半日 主要 非 出 各 河内 ， 及 图 书生 


总 直 地 lreonet | 全 jp 叶 翅 加 用 0 


篇 本 地 Intranet | 各- 椒 式 : 然 用 hh 0 
图 4.6 应 用 str_replace0 函 数 蔡 换 子 字符 串 图 4.7 应 用 str_ireplace0 函 数 对 查询 关键 字 描 红 


2. substr_replace() 函 数 
对 指定 字符 串 中 的 部 分 字符 串 进行 蔡 换 。 


ed substr_replace ( string str,string repl, int start,[int length]) 
参数 说 明 如 表 4.5 所 示 。 


表 4.5 substr_replace() 函 数 的 参数 说 明 


参数 说 明 

str 指定 要 操作 的 原始 字符 串 

repl 指定 葵 换 后 的 新 字符 串 

i 指定 葵 换 字符 串 开始 的 位 置 。 正 数 表示 起 始 位 置 从 字符 串 开头 开始 :负数 表示 起 
始 位 置 从 字符 串 的 结尾 开始 ; 0 表示 起 始 位 置 字符 串 中 的 第 一 个 字符 

jan 可 选 参数 ， 指 定 蔡 换 的 字符 串 长 度 。 默 认 值 是 整个 字符 串 。 正 数 表 示 起 始 位 置 从 


字符 串 开 头 开始 ; 负数 表示 起 始 位 置 从 字符 串 的 结尾 开始 ; 0 表示 插入 而 非 蔡 代 


et length 数值 小 于 或 等 于 start 数值 ， 那 么 length 的 值 自动 为 0。 


例 4.17 下 面 将 指定 字符 串 中 的 “ 双 倍 ” 蔡 换 为 “百倍 ”， 并 且 输 出 蔡 换 后 的 结果 。 (实例 位 置 : 光 
盘 \TM\Instance\04\4.17) 

实例 代码 如 下 : 

<?php 

$str=" 用 今日 的 辛勤 工作 ， 换 明日 的 双 倍 回报 !"; // 定 义 字符 串 常量 


局 
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Sreplace=" 百 倍 "; /定义 要 替换 的 字符 串 
echo substr_replace($str,$replace,26.4); // 蔡 换 字符 串 
Ee 


结果 为 : 用 今日 的 辛勤 工作 ， 换 明日 的 百倍 回报 ! 


4.11 什么 是 正则 表达 3 


正则 表达 式 是 一 种 描述 字符 串 结构 的 语法 规则 ， 是 一 个 特定 的 格式 化 模式 。 它 可 以 匹配 、 蔡 换 、 截 取 
匹配 的 字 串 。 对 于 用 户 来 说 ， 可 能 以 前 接触 过 DOS， 如 果 想 匹配 当前 文件 夹 下 所 有 的 文本 文件 ， 可 以 输入 
“dir *.txt” 命 令 ， 按 Enter 键 后 ， 所 有 “.txt” 文 件 将 会 被 列 出 来 。 这 里 的 “*.txt” 就 可 以 理解 为 一 个 简单 的 
正则 表达 式 。 

在 学 习 正则 表达 式 前 ， 我 们 先 来 了 解 一 个 正则 表达 式 中 的 几 个 容易 混淆 的 术语 ， 这 对 于 学 习 正 则 表达 
式 有 很 大 的 帮助 。 

回 grep: 最 初 是 ED 编辑 器 中 的 一 条 命令 ,用 来 显示 文件 中 特定 的 内 容 。 后 来 成 为 一 个 独立 的 工具 grep。 

egrep: grep 虽然 不 断 地 更 新 升级 ， 但 仍然 无 法 跟 上 技术 的 脚步 。 为 此 ， 贝 尔 实验 室 写 出 了 egrep， 
意 为 “扩展 的 grep”。 这 大 大 增强 了 正则 表达 式 的 能 力 。 

POSIX (Portable Operating System Interface of UNIX) : 可 移植 操作 系统 接口 。 在 grep 发 展 的 同时 ， 
其 他 一 些 开 发 人 员 也 根据 自己 的 喜好 开发 出 了 具有 独特 风格 的 版 本 。 但 问题 也 随 之 而 来 。 有 的 程 
序 支持 某 个 元 字符 ， 而 有 的 程序 则 不 支持 。 因 此 ， 就 有 了 POSIX。POSIX 是 一 系列 标准 ， 确 保 了 
操作 系统 之 间 的 移植 性 。 不 过 POSIX 和 SQL 一 样 ， Oe a el A i -个 参考 。 

了 Perl (Practical Extraction and Reporting Language) : 实际 抽取 与 汇报 语言 。1987 年 ，Larry Wall 发 
布 了 Perl。 在 随后 的 7 年 时 间 里 ， 从 Perll 到 现在 的 Perl5， 最 终 成 为 了 posIX 之 后 的 另 一 个 标准 。 
PCRE: Perl 的 成 功 ， 让 其 他 的 开发 人 员 在 某 种 程度 上 要 兼容 “Perl”， 包 括 C/C++、Java、Python 
等 都 有 自己 的 正则 表达 式 。1997 年 ，Philip Hazel 开发 了 PCRE 库 ， 这 是 兼容 Perl 正则 表达 式 的 一 
套 正则 引擎 ， 其 他 开发 人 员 可 以 将 PCRE 整合 到 自己 的 语言 中 ， 为 用 户 提供 丰富 的 正则 功能 。 许 
多 软件 都 使 用 PCRE，PHP 正 是 其 中 之 


加 


[al 


[al 


4.12 正则 表达 式 语 法 规则 


医 1 视频 讲解 : 光盘 \TM\Video\ 第 4 章 \ 视 频 4.12\ 正 则 表达 式 语法 规则 .exe 

正则 表达 式 由 两 部 分 构成 :元 字符 和 文本 字符 。 元 字符 就 是 具有 特殊 含义 的 字符 ， 如 前 面 提 到 的 “*” 
和 “? ”。 文 本 字符 就 是 普通 的 文本 ， 如 字母 和 数字 等 。PCRE 风格 的 正则 表达 式 一 般 都 放置 在 定 界 符 “/” 
中 间 ， 如 “Aw+([-+.]w+)*@\Ww+([-.] 人 w+ NWw+([-.] 人 w+)*/” 、“/^http:VV(www\.)?.+.?8/”。 为 了 便于 读者 理 
解 ， 除 个 别 实例 外 ， 本 节 中 的 表达 式 不 给 出 定 界 符 “/”。 


4.12.1 行 定 位 符 〈^ 和 9$) 
行 定位 符 用 于 描述 字 串 的 边界 。“^” 表 示 行 的 开始 : “$” 表 示 行 的 结尾 。 例 如 : 
Amingri 


该 表达 式 表示 要 匹配 字 串 mr 的 开始 位 置 是 行头 ， 如 “mrsoft”、“mrbook” 都 可 以 匹配 ,而 “Tomorrow 
mr” 则 不 能 匹配 。 如 果 要 匹配 “Tomorrow mr”， 则 可 以 使 用 下 面 的 行 结尾 定位 符 “$”。 
@ 
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mingri$ 

这 可 以 匹配 以 mingri 结尾 的 字符 ， 而 不 能 匹配 以 mingri 开头 的 字符 ， 如 果 要 匹配 的 字 串 可 以 出 现在 字 
符 串 的 任意 部 分 ， 那 么 可 以 直接 写成 : 

mingri 


这 样 两 个 字符 串 就 都 可 以 匹配 了 。 
4.12.2 字符 类 〈[]) 


正则 表达 式 是 区 分 大 小 写 的 ， 如 果 想 要 忽略 大 小 写 ， 可 以 通过 方 括号 ([]) 表达 式 来 完成 。 只 要 匹配 的 
字符 出 现在 方 括号 内 ， 即 表示 匹配 成 功 。 但 需要 注意 的 是 ， 一 个 方 括号 只 能 匹配 一 个 字符 。 例 如 ， 要 匹配 
字 串 “mingri” 不 区 分 大 小 写 ， 其 表达 式 的 格式 如 下 。 

[MINGming][Riri] 

这 样 ， 就 可 以 匹配 字 串 “mingri” 的 所 有 写法 。POSIX 和 PCRE 都 使 用 了 一 些 预定 义 字符 类 ， 但 表示 方 
法 略 有 不 同 。POSIX 风格 的 预定 义 字 符 类 如 表 4.6 所 示 。 


表 4.6 POSIX 预定 义 字符 类 


预定 义 字符 类 说 明 
:digit: 十 进 制 数 字 集 合 。 等 同 于 [0-9] 
:alnum: 字母 和 数字 的 集合 。 等 同 于 [a-zA-Z0-9] 
:alpha: 字母 集合 。 等 同 于 [a-zA-Z] 
:blank: 空格 和 制 表 符 
:xdipgit: 十 六 进 制 数字 
:punct: 特殊 字符 集合 。 包 括 键 盘 上 的 所 有 特殊 字符 ， 如 !1@Q#$? 等 
:print: 所 有 的 可 打印 字符 (包括 空白 字符 ) 
:space: 空白 字符 空格 、 换 行 符 、 换 页 符 、 回 车 符 、 水 平 制 表 符 ) 
:graph: 所 有 的 可 打印 字符 〈 不 包括 空白 字符 
:upper: 所 有 大 写字 母 。[A-Z] 
‘lower: 所 有 小 写字 母 。[a-z] 
:cntrl: 控制 字符 


而 PCRE 的 预定 义 字符 类 则 是 使 用 反 斜 杠 来 表示 的 。 
4.12.3 选择 字符 〈|) 


要 忽略 字 串 的 大 小 写 ， 还 可 以 通过 选择 字符 〈|) 来 完成 。 该 字符 可 以 理解 为 “或 ”， 如 上 例 也 可 以 写成 : 
MImRIr 
该 表达 式 的 意思 是 以 字母 M 或 m 开头 ， 后 面 接 一 个 字母 RR 或 7。 


of 
Di “D” 和 使 用 “]?” 的 区 别 在 于 : “0” 只 能 匹配 单个 字符 ， 而 “|” 可 以 匹配 任意 长 度 的 字 囊 。 


4.12.4” 连 字符 (-) 


变量 的 命名 规则 是 只 能 以 字母 和 下 划 线 开头 。 如 果 要 使 用 正则 表达 式 来 匹配 变量 名 的 第 一 个 字母 ， 要 


@ 
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写成 如 下 格式 。 

[ab,cd…A,B,C.D… ] 

这 实在 是 非常 麻烦 ， 但 正则 表达 式 提供 了 连 字符 “-” 来 解决 这 个 问题 。 连 字符 可 以 表示 字符 的 范围 。 
如 上 例 可 以 写成 : 

[a-zA-Z] 


4.12.5 “排除 字符 〈 准 ) 


上 面 的 例子 是 匹配 符合 命名 规则 的 变量 。 反 过 来 ， 匹 配 不 符合 命名 规则 的 变量 。 正 则 表达 式 提供 了 “^” 
字符 。 这 个 元 字符 在 4.12.1 节 中 曾经 出 现 过 ， 表 示 行 的 开始 。 这 里 将 其 放 到 方 括号 中 ， 则 表示 排除 的 意思 。 
例如 : 

[‘a-zA-Z ] 

该 表达 式 匹配 的 是 不 以 字母 和 下 划 线 开头 的 变量 名 。 


4.12.6 ”限定 符 (? * + {n,m}) 


经 常 使 用 google 的 用 户 可 能 会 发 现 ， 在 搜索 结果 页 的 下 方 ，“google ”中间 字母 “o” 的 个 数 会 随 着 搜 
索 页 的 改变 而 变化 。 那 么 要 匹配 该 字 串 的 正则 表达 式 该 如 何 实现 呢 ? 
对 于 这 类 重复 出 现 的 字母 或 字 串 ， 可 以 使 用 限定 符 来 实现 匹配 。 限 定 符 主要 有 6 种 ， 如 表 4.7 所 示 。 
表 4.7 限定 符 的 说 明和 举例 


g00°.gle 
employe{0.2} ， 该 表达 式 可 以 匹配 employ 、employe 和 
employee 3 种 情况 


通过 对 google 应 用 的 观察 发 现 ， 当 搜索 结果 只 有 一 页 时 ， 不 显示 google 标志 ; 只 有 大 于 等 于 2 时 ， 才 
显示 google。 这 说 明 字 母 “o” 最 少 为 两 个 ， 而 最 多 可 以 看 到 20 个 ， 那 么 其 正则 表达 式 为 : 
go{2,20}gle 


4.12.7 点 字符 (.) 


在 正则 表达 式 中 通过 点 字符 “.” 可 以 匹配 除 换行 符 外 的 任意 一 个 字符 。 注 意 ， 是 除 换行 符 外 的 任意 的 
一 个 字符 。 
例 4.18 在 本 例 中 应 用 正则 表达 中 的 点 字符 (.〉 匹 配 变量 中 是 否 包 含 指定 的 字符 串 。〔 实 例 位 置 ， 光 
盘 \IM\Instance\04\4.18) 
程序 代码 如 下 : 
<?php 
$str=" 吉 林 省 明日 科技 有 限 公 司 "; 
if(preg_match("/^ 阴 日 .科技 $/",$str,$counts)X{ /判断 变量 中 是 否 包含 明日 、 科 技 字符 串 
echo "<script>alert( 没 有 指定 的 字符 串 !"); history.back();</script>"; 
® 
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Jelse{ 


} 


?> 


echo "<script>alert(' 存 在 指定 的 字符 串 !"); history.back();</script>"; 


结果 为 : 存在 指定 的 关键 字 ! 


4.12.8 反 斜 杠 〈\) 


除了 可 以 做 转 义 字符 外 ， 反 斜 本 还 有 其 他 一 些 功能 。 
反 斜 杠 可 以 将 一 些 不 可 打印 的 字符 显示 出 来 ， 如 表 4.8 所 示 。 


表 4.8 反 和 斜 杠 显示 的 不 可 打印 字符 


字符 说 明 
警报 ， 即 ASCII 中 的 <BEL> 字 符 (0x07) 
\b 退 格 ， 即 ASCII 中 的 <BS> 字 符 〈0x08) 。 注 意 ， 在 PHP 中 只 有 在 中 括号 ([ ]) 里 使 用 才 表 示 退 格 
\e escape， 即 ASCI 中 的 <ESC> 字 符 〈0x1B) 
¥ 换 页 符 ， 即 ASCIT 中 的 <FF> 字 符 〈0x0C) 
un 换行 符 ， 即 ASCI 中 的 <LF> 字 符 〈0x0A) 
回 车 符 ， 即 ASCI 中 的 <CR> 字 符 〈0x0D) 
Vt 水 平 制 表 符 ， 即 ASCI 中 的 <HT> 字 符 (0x09) 
\xhh 十 六 进 制 代码 
\ddd 八进制 代码 
\cX 即 control-x 的 缩写 ， 匹 配 由 x 指明 的 控制 字符 ， 其 中 x 是 任意 字符 


还 可 以 指定 预定 义 字符 集 ， 如 表 4.9 所 示 。 


表 4.9 反 和 斜 杠 指定 的 预定 义 字符 集 


限 定 符 说 明 
\d 任意 一 个 十 进 制 数字 ， 相 当 于 [0-9] 
D 任意 一 个 非 十 进 制 数字 
's 任意 一 个 空白 字符 〈 空 格 、 换 行 符 、 换 页 符 、 回 车 符 、 水 平 制 表 符 ) ， 相 当 于 [fed 
\S 任意 一 个 非 空白 字符 
Ww 任意 一 个 单词 字符 ， 相 当 于 [a-zA-Z0-9 ] 
\W 任意 一 个 非 单词 字符 


反 斜 杠 还 有 一 种 功能 ， 就 是 定义 断言 ， 其 中 已 经 接触 过 了 \b、\B， 其 他 如 表 4.10 所 示 。 


表 4.10 反 和 斜 杠 定义 断言 的 限定 符 


限 定 符 说 明 
vb 单词 分 界 符 ， 用 来 匹配 字符 串 中 的 菜 些 位 置 ，\b 是 以 统一 的 分 界 符 来 匹配 
B 非 单词 分 界 符 序列 
从 总 是 能 够 匹配 待 搜索 文本 的 起 始 位 置 
表示 在 未 指定 任何 模式 下 匹配 的 字符 , 通常 是 字符 串 的 末尾 位 置 , 或 者 是 在 字符 串 末尾 的 换行 符 之 
前 的 位 置 
忆 只 匹配 字符 串 的 末尾 ， 而 不 考虑 任何 换行 符 


当前 匹配 的 起 始 位 置 
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4.12.9 反 向 引用 


反 向 引用 ,就 是 依靠 子 表达 式 的 “记忆 ”功能 来 匹配 连续 出 现 的 字 串 或 字母 。 例 如, 匹配 连续 两 个 “it”， 


首先 将 单词 “it” 作 为 分 组 ， 然 后 在 后 面 加 上 “\1” 即 可 。 格 式 为 : 


式 。 


(OM 
这 就 是 反 向 引用 最 简单 的 格式 。 如 果 要 匹配 的 字 串 不 固定 ， 那 么 就 将 括号 内 的 字 串 写成 一 个 正则 表达 
如 果 使 用 了 多 个 分 组 ， 那 么 可 以 用 “\1”、“\2” 来 表示 每 个 分 组 (顺序 是 从 左 到 右 〉。 例 如 : 
([a-z])([A-ZI)\1\2 

除了 可 以 使 用 数字 来 表示 分 组 外 ， 还 可 以 自己 指定 分 组 名 称 。 格 式 为 : 

(?P<subname>…) 

如 果 想 要 反 向 引用 该 分 组 ， 使 用 如 下 语法 。 

(?P=subname) 

下 面 重 写 一 下 表达 式 ([a-z])([A-Z])N1\2， 为 这 两 个 分 组 分 别 命名 ， 并 反 向 引用 它们 。 正 则 表达 式 如 下 。 
(?P<fir>[a-z])(?P<sec>[A-Z])(?P=fir)(?P=sec) 


4.13 ” POSIX 扩展 正则 表达 式 函 数 


殴 il 视频 讲解 :光盘 \TM\Video\ 第 4 章 \ 视 频 4.13\POSIX 扩展 正则 表达 式 函 数 .exe 
PHP 中 实现 POSIX 正则 表达 式 的 函数 有 7 个 。 下 面 就 来 了 解 其 中 几 个 常用 函数 。 


4.13.1 替换 字符 串 


函数 格式 : 
string ereg_replace/eregi_replace ( string pattern, string replacement, string string ) 
函数 功能 : 在 字符 串 string 中 匹配 表达 式 pattem。 如 果 匹 配 成 功 ， 则 使 用 replacement 来 蔡 换 匹 配 字 串 ， 并 


返回 替换 后 的 string。 如 果 未 在 string 中 找到 匹配 项 ， 则 string 将 原样 返回 。eregi_replace0 函 数 不 区 分 大 小 写 。 


例 4.19 将 字符 串 中 所 有 非 大 写 的 “mingri” 都 换 成 大 写 “MINGRI”。( 实 例 位 置 : 光盘 \TMNInstance\04\4.19) 
实例 代码 如 下 : 


<?php 
S$ereg = 'mingri'; // 要 匹配 的 字 串 
$str = 'nihao, MINGri,mingri,mingRIl.'; // 要 查找 的 文本 
Srep_str = eregi_replace($ereg,'MINGRI', $str); /使 用 eregi_replace() 函 数 进行 替换 
echo $rep_str; /| 输出 替换 后 的 文本 
?> 


结果 为 : nihaoMINGRLMINGRLMINGRI. 


4.13.2 分割 字符 串 


函数 格式 : 
array split/spliti ( string pattern, string string [, int limit] ) 
函数 功能 : 使 用 表达 式 pattern 来 分 割 字符 串 string。 如 果 有 参数 limit， 那 么 数组 最 多 有 limit 个 元 素 ， 


剩余 部 分 都 写 到 最 后 一 个 数组 元 素 中 。 如 果 函 数 错误 ， 则 返回 false。split0 函 数 区 分 大 小 写 ， 而 spliti0 函 数 


四 
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不 区 分 大 小 写 。 
例 4.20 通过 spliti0 函 数 ， 使 用 字 串 “、” 来 分 割 字符 串 $str。 (实例 位 置 : 光盘 \TMNImstance\04\4.20) 
代码 如 下 : 
<?php 
$ereg="、" // 分 割 字符 串 使 用 的 表达 式 
$str = JAVA 编程 词典 、VB 编程 词典 、VC 编程 词典 、PHP 编程 词典 ; ”// 要 被 分 割 的 字符 串 
Sarr_str = spliti($ereg, $str); /使 用 spliti() 函 数 分 割 字符 串 
var_dump($arr_str); /显示 分 割 后 的 数组 结构 
?> 


结果 为 : array(4) { [0]=> string(15) "PHP 编程 词典 " [1]=> string(13) "C 编程 词典 " [2]=> string(15) "ASP 
编程 词典 " [3]=> string(16) "JAVA 编程 词典 " }。 


4.14 PCRE 兼容 正则 表达 式 函 数 


名 视频 讲解 : 光盘 \TM\Video\ 第 4 章 \PCRE 兼容 正则 表达 式 函 数 .exe 
实现 PCRE 风格 的 正则 表达 式 的 函数 也 有 7 个 。 但 无 论 从 执行 效率 ， 还 是 从 语法 支持 上 ，PCRE 函数 都 
要 略 优 于 POSIX 函数 。 下 面 对 其 中 几 个 常用 的 PCRE 函数 进行 讲解 。 


4.14.1 查找 字符 串 


函数 格式 : 
int preg_match/preg_match_all ( string pattern, string subject [, array matches] ) 
函数 功能 : 在 字符 串 subject 中 匹配 表达 式 pattem。 函 数 返 回 匹 配 的 次 数 。 如 果 有 数组 matches， 那 么 每 
匹配 的 结果 将 被 存储 到 数组 matches 中 。 
函数 preg_matchO 的 返回 值 是 0 或 1。 因 为 该 函数 在 匹配 成 功 后 就 停止 继续 查找 了 。 而 preg_match_all0 函 
数 则 会 一 直 匹配 到 最 后 才 停止 。 参 数 array matches 对 于 preg_match_all0 函 数 是 必需 的 ， 而 对 前 者 则 可 以 省 略 。 
例 4.21 使 用 preg_matchO0 和 preg_match_all0 函 数 验证 手机 和 座机 号 码 的 格式 是 否 正确 , 并 返回 各 自 的 
匹配 次 数 。〔 实 例 位 置 ， 光盘 \TM\Instance\04\4.21) 
实例 代码 如 下 : 
<?php 
$str='this is a mingribook luntan xiaoxi' 
S$preg="\b\W{6}\b/"; 
Snum1=preg_match($preg,$str, $str1); 
echo $num1.'<br>"; 
var_dump($str1); 
$num2 =preg_match_all($preg,$str,$str2); 
echo '<p>"'.$num2.'<br>"; 
var_dump($str2); 


党 


?> 


运行 结果 如 图 4.8 所 示 。 
4.14.2 ”替换 字符 串 


函数 语法 : 


@ 


图 4.8 preg_ matchO0 和 preg_match all0 函 数 
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mixed preg_replace ( mixed pattern, mixed replacement, mixed subject [, int limit] ) 

函数 功能 : 该 函数 在 字符 串 subject 中 匹配 表达 式 pattem， 并 将 匹配 项 蔡 换 成 字 串 replacement。 如 果 有 
参数 limit， 则 蔡 换 limit 次 。 

例 4.22 ”将 输入 的 “[b].…[b]”、“ 国 .向 ”等 类 似 的 格式 转换 为 html 能 识别 的 标签 。 (实例 位 置 ; 光 
盘 \TMNVInstance\04\4.22) 

实例 代码 如 下 : 


<?php 

Sstring = '[b] 粗 体 字 [b]'; 

$b_rst = preg_replace('N\[b\](.*)\[Vb\/i','<b>$1</b>",$string); 
echo $b_rst; 

= 


结果 为 : 粗 体 字 。 


of 
和 培 明 Preg_replace() 函 数 中 的 字 串 “$1” 是 在 正则 表达 式 外 调用 分 组 ， 按 照 S1、$2 排列 ， 依 次 表示 
从 左 到 右 的 分 组 顺序 ， 也 就 是 括号 顺序 。 另 外 ，$0 表示 的 是 整个 正则 表达 式 的 匹配 值 。 


4.15 实 战 
坎 4 视频 讲解 :光盘 \TM\Video\ 第 4 章 \ 实 战 .exe 


4.15.1 超 长 文本 的 分 页 显示 


例 4.23 ”本 实例 实现 超 长 文本 的 分 页 输出 , 其 中 通过 strlen0 函 数 获取 字符 串 的 长 度 , 应 用 ceil0 函 数 将 文 
本 文件 中 的 数据 分 页 ， 通 过 自 定义 函数 msubstr0 控 制 数据 的 输出 ， 通 过 substr0 函 数 获取 到 当前 页 面 输出 的 
内 容 。〔 实 例 位置 ， 光盘 \TM\Instance\04\4.23) 

在 程序 开发 的 过 程 中 ， 经 常会 遇 到 输出 超 长 文本 文件 的 情况 ， 为 了 使 页 面 的 布局 更 加 合理 ， 文 本 文件 
的 输出 更 加 流 收 ， 采 用 分 页 输出 文本 文件 是 一 个 很 好 的 方法 。 运 行 本 实例 ， 如 图 4.9 所 示 ， 通 过 分 页 来 输出 
超 长 文本 文件 。 


全 的 私人 号 本 专家 


Ml FE 


图 4.9 新 闻 主题 信息 显示 
(1) 编写 function.php 文件 ， 创 建 自 定义 函数 unhtml0 对 超 长 文本 中 的 特殊 字符 进行 蔡 换 ， 创建 自 定义 
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函数 msubstrO 控 制 超 长 文本 中 数据 的 输出 。 语 法 如 下 : 


<?php 
function unhtml($content}{ /定义 自 定义 函数 的 名 称 
S$content=htmlspecialchars($content); /| 转换 文本 中 的 特殊 字符 
S$content=str_replace(chr(13),"<br>",$content); // 艾 换 文本 中 的 换行 符 
$content=str_replace(chr(32),"&nbsp:",$content); // 蔡 换文 本 中 的 &nbsp; 
S$content=str_replace("L[","<",$content]; /车 换文 本 中 的 大 于 号 
$content=str_replace(")_)",">",$content); /车 换文 本 中 的 小 于 号 
S$content=str_replace("|_|"," ",$content); /| 蔷 换 文本 中 的 空格 
return trim($content); /删除 文本 中 首尾 的 空格 
} 
function msubstr($str,$start,$len){ //$str 指 的 是 字符 串 ，$start 指 的 是 字符 串 的 起 始 位 置 ，$len 指 的 是 长 度 
$strlen=$start+ $len; /用 $strlen 存储 字符 串 的 总 长 度 (从 字符 串 的 起 始 位 置 到 字符 串 的 总 长 度 ) 
for($i=0;$i<$strlen;$i++\{ // 通 过 for 循环 语句 ， 循 环 读 取 字符 串 
iflord(substr($str,$i,1))>0xa0){ /如 果 字 符 串 中 首 个 字 节 的 ASCII 序数 值 大 于 0xa0, 则 表示 为 汉字 
$tmpstr.=substr($str,$i,2); /每 次 取出 两 位 字符 赋 给 变量 $tmpstr， 等 于 一 个 汉字 
Si++; /变量 自 加 1 
}else{ // 如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 $tmpstr 
Stmpstr.=substr($str,$i,1); 
: 
return $tmpstr; // 输 出 字符 串 
8 
?> 


2) 创建 index.php 页 面 ， 首 先 定 义 分 页 变量 page， 通 过 include 包含 语句 调用 function.php 文件 。 然 
后 通过 fle_get_contents(O) 函 数 读 取 根 目录 下 file 文件 夹 中 file.txt 文件 中 的 数据 ， 实 现 数据 的 分 页 输出 。 其 关 
键 代 码 如 下 : 

<?php if ($page==") {$page=1;}; include("function.php");?> 


<?php 
// 读 取 超 长 文本 中 的 数据 ， 实 现 超 长 文本 中 数据 的 分 页 显示 


if($page}{ 
$counter=file_get_contents("file/file.txt"); // 读 取 指 定 文件 夹 下 的 文件 
S$length=strlen(unhtml($counter)); // 获 取 数 据 的 长 度 ， 应 用 unhtml() 函 数 去 除 特殊 字符 
$page_count=ceil($length/950); /计算 共有 多 少 页 
$c=msubstr($counter,0,($page-1)*950); // 获 取 上 一 页 的 数据 
$c1=msubstr($counter,0,$page*950); // 获 取 当 前 页 的 数据 
echo substr($c1,strlen($c),strlen($c1)-strlen($c)); /输出 当前 页 的 数据 

a 

(3) 设置 分 页 使 用 的 超 链接 ， 定 义 链接 的 标识 。 其 代码 如 下 : 
<tr> 


<td width="202” height="22"><span class="STYLE3"> 页 次 : <?php echo S$page;?> / <?php echo 
$page_count'?> 页 </span></td> 
<td width="278"><span class="STYLE3"> 分 页 : 
<?php 
if($page!=1X{ 
echo "<a href=index.php?page=1> 首 页 </a>&nbsp;"; 
echo "<a href=index.php?page=".($page-1)." > 上 一 页 </a>&nbsp;"; 


} 
if($page<$page_count}{ 


echo "<a href=index.php?page=".($page+1)."> 下 一 页 </a>&nbsp;"; 
echo "<a href=index.php?page=".$page_count."> 尾 页 </a>"; 


© 
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?> 
</span></td> 
</tr> 


4.15.2 ”控制 页 面 中 输出 字符 串 的 长 度 


例 4.24 ”本 实例 应 用 sttlen0 函 数 获 取 字 符 串 的 长 度 ， 在 输出 字符 串 时 进行 判断 ， 如 果 字 符 串 超出 指定 
的 长 度 ， 则 使 用 指定 的 字符 进行 蔡 换 ， 并 在 输出 时 间 字 符 串 时 ， 应 用 str_replace0 函 数 将 “-” 蔡 换 为 “/”。 
(实例 位 置 ， 光盘 \TM\Instance\04\4.24) 

在 论坛 或 者 电子 商务 等 网 站 中 ， 经 常会 输出 一 些 公告 信息 、 最 新 动态 等 内 容 ， 这 些 内 容 都 是 以 标题 的 
形式 进行 输出 ， 为 标题 设置 超 链接 ， 链 接 到 相关 内 容 的 详细 信息 页 面 。 

在 输出 标题 信息 时 ， 由 于 要 考虑 页 面 规范 化 、 设 计 合 理 ， 所 以 要 对 输出 的 标题 长 度 进行 限制 ， 如 果 标 
题 的 总 长 度 超出 指定 范围 ， 就 需要 使 用 省 略 号 进行 蔡 换 。 运 行 结果 如 图 4.10 所 示 。 


入 了 为 区) 
区 


未 RE) 
各 于 。 计 咎 枯 利 罕 与 拷 本。 从 者 市 二 并 


4.10 ”控制 页 面 中 输出 字符 串 的 长 度 


(1) 创建 index.php 页 面 ， 首 先 通过 include_once 语句 调用 数据 库 连接 文件 和 字符 串 处 理 文件 ， 然 后 执 
行 查询 语句 ， 查 询 出 数据 表 中 的 记录 ， 最 后 输出 最 新 动态 的 标题 和 发 布 时 间 ， 并 判断 如 果 标 题 的 内 容 超过 
24 个 字 节 ， 则 输出 省 略 号 ， 截 取 发 布 时 间 。 其 关键 代码 如 下 : 
<?php 
include_once("conn/conn.php"); // 调 用 连接 数据 库 的 文件 
include_once("function.php"); 


?> 
<table width="365" height="22" border="0" align="center" cellpadding="0" cellspacing="0"> 
<?php 
$sql=mysql_query("select * from student order by age desc ",$id); 
while($myrow=mysql_fetch_array($sql)X{ 


<tr> 
<td width="20" height="22"><div align="center"><img src="images/01.JPG"/></div></td> 
<td width="258" height="22"><a href="new_dynamic.php?id=<?php echo $myrow["id"];?>"> 
<?php 
echo unhtml(msubstr($myrow["name"],0,24)); 
echo '&nbsp;&nbsp;&nbsp;"; 
echo unhtml(msubstr($myrow["zhuanye"],0,24)); 


?> 


® 


PHP+MySQL 开发 实战 


?> 
</a> </td> 
<td width="87"> 
<?php 
echo"<fontcolor=red> (".substr(str_replace("-","/", $myrow[address]),0,10).") </font>"; 
?> 
</td> 
<ltr> 
<tr><td colspan="3"></td></tr> 
<?php } ?> 
</table> 
(2) 创建 new_dynamic.php 页 面 ， 根 据 超 链接 中 传递 的 ID 值 ， 从 数据 表 中 查询 出 指定 的 记录 ， 并 输出 
记录 的 详细 内 容 。 其 关键 代码 如 下 : 
<?php 
include_once("conn/conn.php"); // 调 用 连接 数据 库 的 文件 
include_once("function.php"); 
?> 
<?php 
$sql=mysql_query("select * from student where id=".$_GETI[id]."",$id); 
while($myrow=mysql_fetch_array($sql)X{ 
Ss 
<td align="left" valign="top" class="STYLE1"><span class="STYLE1"> 
<?php echo $myrowIname];?> <br><br> 
<?php echo $myrow[zhuanye];?></span></td> 
<?php }?> 


4.15.3 ”正则 无 刷新 用 户 注册 


例 4.25 ”通过 正则 表达 式 无 刷新 验证 用 户 注册 信息 是 否 合理 。 正 则 表达 式 经 常 被 应 用 于 对 用 户 注册 信息 的 
合理 性 判断 中 ， 可 以 对 用 户 输 入 的 邮编 、 电 话 号 码 、 邮 箱 地 址 和 网 址 的 格式 进行 判断 。 本 实例 中 应 用 正则 表达 
式 和 JavaScript 脚本 ， 实 现 无 刷新 判断 用 户 输入 信息 的 格式 是 否 正 确 。〔 实 例 位 置 ， 光盘 \TM\Instance\04\4.25) 

(1) 本 例 中 将 正则 表达 式 定义 到 自 定义 函数 中 ， 然 后 在 JavaScript 脚本 中 调用 自 定义 函数 ， 执 行 对 用 
户 输入 信息 的 判断 。JavaScript 脚本 文件 checkjs 的 代码 如 下 : 
function checkregtel(regtel}{ // 通 过 正则 表达 式 验证 手机 号 码 的 格式 是 否 正确 
var str=regtel; 
var Expression=/^13(\d{9})$|^15(\d{9})$/; // 定 义 正则 表达 式 
Var objExp=new RegExp(Expression); 
if(objExp.test(str)==trueX{ 
return true; 
}else{ 
return false; 
} 


function checkregtels(regtels)f // 自 定义 函数 ， 验 证 座机 号 码 的 格式 是 否 正确 
Var str=regtels; 
Var Expression=/^(\d{3}-)(\d{8})$I^(\d{4}-)(d{7})$I^(\d{4}-)(d{8})$/;”// 定 义 正则 表达 式 
Var objExp=new RegExp(Expression); 
if(objExp.test(str)==trueX{ 
return true; 
jelse{ 


return false; 


@_ 
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} 

function checkregemail(emailsX{ // 自 定义 函数 ， 验 证 E-mail 地 址 的 格式 是 否 正确 
var str=emails; 
var Expression=/\w+([-+.JWw+)*@\W+([-.JWw+)"\ w+([-.JWw+)*/; /定义 正则 表达 式 


Var objExp=new RegExp(Expression); 
if(objExp.test(str)==trueX{ 


return true; 
Jelse{ 
return false; 
} 
, 
function chkreginfo(form,mark){ // 自 定义 函数 ， 对 表单 中 用 户 提交 的 数据 进行 判断 
// 这 里 省 略 了 部 分 代码 
if(mark==4 || mark=="all"){ // 验 证 E-mail 地 址 的 格式 是 否 正确 

if(form.email.value=="™"){ // 如 果 输 入 值 为 空 
chknew_email.innerHTML=" 请 输入 E-mail 地 址 ! "; // 则 返回 “请 输入 E-mail 地 址 ! ” 
form.email.style.backgroundColor="#FF0000"; /定义 返回 提示 信息 的 样式 
return false; 
jelse if(!checkregemail(form.email.value)}{ // 调 用 自 定 义 函 数 验 证 E-mail 地 址 格式 

chknew_emailinnerHTML=" 邮 箱 地 址 的 格式 不 正确 ! "; 
form.email.style.backgroundColor="#FF0000"; 
return false; 
Jelse{ 
chknew_email.innerHTML="™"; // 当 返回 值 是 空 时 
form.email.style.backgroundColor="#FFFFFF"; /定义 样式 为 白色 
if(mark==5 || mark=="all")f // 验 证 手机 号 码 的 格式 
if(form.mtel.value=="™"){ 
chknew_mtel.innerHTML=" 请 输入 电话 号 码 !"; 
form.mtel.style.backgroundColor="#FF0000"; 
return false; 

}else if(Icheckregtel(form.mtel.value)}{ /调用 自 定义 函数 验证 手机 号 码 格式 是 否 正确 
chknew_mtel.innerHTML=" 电 话 号 码 的 格式 不 正确 !"; 
form.mtel.style.backgroundColor="#FF0000"; 
return false; 

}else if(isNaN(form.mtel.value)}X{ 
chknew_mtel.innerHTML=" 电 话 号 由 数字 组 成 !"; 
form.mtel.style.backgroundColor="#FF0000"; 
return false; 

}else{ 
chknew_mtel.innerHTML="™"; 
form.mtel.style.backgroundColor="#FFFFFF"; 

} 

} 
/省 略 了 部 分 代码 


} 
(2) 正则 表达 式 和 JavaScript 脚本 文件 创建 完成 后 ， 接 下 来 开始 创建 index.php 文件 ， 添 加 form 表单 ， 
调用 JavaScript 脚本 文件 ， 提 交 用 户 注册 信息 。 其 关键 代码 如 下 : 
<link rel="stylesheet" type="text/css" href= "css/style.css"> 
<script src="js/check.js"></script> 
<form name="form_reg" method="post" action="index_ok.php" onSubmit="return chkreginfo(form_reg,'all)"> 
<table width="620" height="262" border="0" align="center" cellpadding="0" cellspacing="0"> 
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<tr><td width="120" height="30"><div align="right"> 用 户 名 : </div></td> 
<td colspan="3">&nbsp; 
<input type="text" name="recuser" size="20" class="inputcss" onBlur="chkreginfo(form_reg,0)"> 
<div id="chknew_recuser style="color:#FF0000"></div> 
</td> 
</tr> 
<!- 省 略 了 部 分 代码 -> 
<tr><td height="30"><div align="right">E-mail: </div></td> 
<td height="30" colspan="3">&nbsp; 
<input type="text" name="email" size="20" class="inputcss" onBlur="chkreginfo(form_reg,4)"> 
<div id="chknew_email" style="color:#FF0000"></div> 
</td> 
<htr> 
<!- 省 略 了 部 分 代码 --> 
<tr><td height="30">&nbsp;</td> 
<td height="30" colspan="2"> 
<input type="image"” src="images/form (2).jpg"> 
<img src="images/form.jpg" onClick="form_reg.reset()" style="cursor:hand"/> 
</td> 
<ltr> 
</table> 
</form> 


(3) 创建 index_ok.php 文件 ， 输 出 用 户 提交 的 数据 。 运 行 结果 如 图 4.11 所 示 。 
oe mn eins ee oe TE 一 | 
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4.11 通过 正则 表达 式 无 刷新 验证 用 户 注册 信息 


4.15.4 计算 密码 长 度 


CE 


对 密码 的 长 度 进行 计算 ， 以 及 对 密码 长 度 进行 限制 ， 在 实际 应 用 中 比较 广泛 ， 尤 其 是 实现 会 员 注册 功能 。 

例 4.26 本 例 介绍 一 种 应 用 sttlen0 函 数 计算 输入 的 密码 长 度 的 方法 ， 运 行 结果 如 图 4.12 所 示 。 (实例 
位 置 : 光盘 \TM\Instance\04\4.26) 

其 实现 的 过 程 如 下 所 示 。 

(1) 通过 Dreamweaver 设计 一 个 form 表单 ， 表 单元 素 为 一 个 提交 密码 的 文本 域 ， 一 个 “计算 ”按钮 。 

其 代码 如 下 : 

<form action="" method="post" name="form1" class="STYLE1" id="form1"> 

<td width="411" bgcolor="#99CC33"><div align="center"><span class="STYLE2"> 密 码 : 


CC 
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<input name="pass" type="password" size="12" /> 
&nbsp; &nbsp;</span> 
<input type="submit" name="Submit" value=" 计 算 " /> 
</form> 


(2) 在 本 页 中 通过 issetO 函 数 判断 用 户 提交 的 密码 和 按钮 值 是 否 存在 。 如 果 存 在 则 通过 $_ POST0 全 局 


数组 获取 表单 中 提交 的 密码 ， 通 过 strlen0 函 数 计算 出 密码 的 长 度 ， 并 在 页 面 上 显示 。 其 主要 代码 如 下 : 
<?php 
$pass=""; 
ifisset($_POST[pass])&&$_POST[pass]i="){ // 对 form 表单 的 文本 域 进行 判断 
$pass=$_POST[pass]; 
echo strlen($pass); // 通 过 strlen 函数 计算 文本 域 中 的 密码 长 度 


} 


x 


4.15.5 去除 用 户 填 写 注册 信息 中 的 空格 


用 户 在 输入 数据 时 ， 经 常会 在 无 意 中 输 入 多 余 的 空白 字符 ， 在 某 些 情 况 下 ， 字 符 串 中 不 允许 出 现 空白 
字符 和 特殊 字符 , 这 就 需要 去 除 字符 串 中 的 空白 字符 和 特殊 字符 。 在 PHP 中 提供 trim0 函 数 去 除 字符 串 左 右 
两 边 的 空白 字符 和 特殊 字符 ，ltrim0 函 数 去 除 字符 串 左边 的 空白 字符 和 特殊 字符 ，rtrim0 函 数 去 除 字符 串 中 


右边 的 空白 字符 和 特殊 字符 。 


例 4.27 在 本 例 中 应 用 ltrim0 函 数 去 除 用 户 名 和 空格 。 效 果 如 图 4.13 所 示 。 


Instance\04\4.27) 


ifisset($_POST[Submit]) && $_POST[Submit]==" 提 交 "){ 
ifisset($_POST[user]) && $_POST[user]!=") 
S$pass=$_POST[pass']; 
S$tel=$_POSTT'tel]; 
$add=$_POST[add']: 
$str=$_POST[user]; 
Sstrs=ltrim($str)."\n"; 
jelsef 
echo "<script>alert(' 提 交 内 容 不 能 为 空 );</script>"; 


(实例 位 置 ， 光 盘 \TM\ 


// 声 明 关于 文本 域 的 变量 


// 对 表单 的 提交 按钮 进行 判断 
// 对 用 户 名 进行 判断 ， 非 空 时 执行 下 面 的 操作 


// 调 用 用 户 名 文本 框 提交 的 值 
/去 除 用 户 名 左边 的 空格 


EC 


下地 Intranet | 保 j 僧 z 共 必 本 = 


0 而 - 成 100% 一 


图 4.12 计算 密码 长 度 显示 界面 


图 4.13 去除 空格 的 注册 表 
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4.16 小 结 


本 章 主要 讲解 了 字符 串 的 操作 技术 及 正则 表达 式 的 用 法 ， 包 括 通过 单 引 号 、 双 引号 和 定 界 符 标识 字符 
串 ， 字 符 串 的 连接 、 转 义 和 还 原 ， 字 符 串 的 截取 、 比 较 、 检 索 、 蔡 换 和 分 割 等 ， 以 及 正则 表达 式 的 常用 语 
法 规则 ， 并 且 介绍 了 正则 表达 式 的 两 种 主要 风格 ; 另外 ， 通 过 实例 、 实 战 和 实战 练习 对 字符 串 和 正则 表达 
式 的 实际 应 用 进行 了 讲解 。 希 望 通过 本 章 的 学 习 ， 读 者 可 以 对 字符 串 操作 和 正则 表达 式 有 个 初步 的 认识 。 


4.17 学 习 成 果 检验 


1. 尝试 开发 一 个 页 面 ， 验 证 用 户 输入 的 15 位 或 18 位 身份 证 号 长 度 是 否 正确 。 (答案 位 置 ， 光盘 \TM\ 
Instance\04\4.28) 

2. 通过 正则 表达 式 验证 邮箱 地 址 的 格式 是 否 正确 。 (答案 位 置 光盘 \TM\Instance\04\4.29) 

3. 使 用 正则 表达 式 匹 配 html 标签 。 (答案 位 置 : 光盘 \TMNVInstance\04\4.30) 
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外 


初探 数组 


( 名 视频 讲解 。146 分 钟 ) 


数组 是 对 大 量 数 据 进行 有 效 组 织 和 管理 的 手段 之 一 ， 通 过 数组 的 
强大 功能 ， 可 以 对 大 量 数 据 类 型 相同 的 数据 进行 存储 、 排 序 、 插 入 及 
删除 等 操作 ， 从 而 有 效 地 提高 程序 开发 效率 及 改善 程序 的 编写 方式 。 
PHP 是 市 面 上 最 为 流行 的 Web 开发 语言 之 一 ， 具 有 代码 开源 、 升 级 
速度 快 等 将 点 ， 对 数组 的 操作 能 力 更 为 强大 ， 尤 其 为 程序 开发 人 员 提 
供 了 大 量 方便 、 易 懂 的 数组 操作 函数 ， 使 PHP 深 受 广大 Web 开发 人 
员 的 青 睦 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 


H 熟悉 数组 的 类 型 

MW 了 解 如 何 输出 数组 

H 掌握 如 何 遍 历数 组 

WI 掌握 PHP 的 全 局 数组 

Hi 掌握 如 何 统计 数组 元 素 个 数 
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数组 是 一 组 数据 的 集合 ， 把 一 系列 数组 组 织 起 来 ， 
其 他 许多 高 级 语言 中 的 数组 更 灵活 。 数 组 array 是 一 组 有 


序 的 变量 ， 其 中 每 个 变量 称 为 一 个 元 素 。 每 个 元 素 由 一 交 琶 数组 
个 特殊 的 标识 符 来 区 分 , 该 标识 符 称 为 键 (也 称 为 下 标 )， 
变量 中 保存 单个 数据 ， 数 组 中 保存 多 个 变量 ， 如 图 5.1 图 5.1 数组 与 变量 的 关系 
所 示 。 

数组 中 的 每 个 实体 都 包含 两 项 ， 键 和 值 。 其 中 键 可 以 是 数字 、 字 符 串 或 者 数字 和 字符 串 的 组 合 ， 用 于 
标识 数组 中 相应 的 值 ， 而 值 被 称 为 数组 中 的 元 素 ， 可 以 定义 为 任意 数据 类 型 ， 甚 至 是 混合 类 型 。 最 终 通 过 
键 来 获取 相应 的 值 。 例 如 ， 一 个 足球 队 通常 会 有 几 十 个 人 ， 但 是 认识 他 们 时 首先 会 把 他 们 看 作 是 某 队 的 成 
员 ， 然 后 再 通过 他 们 的 号 码 来 区 分 每 一 名 队员 ， 这 时 ， 球 队 就 是 一 个 数组 ， 而 号 码 就 是 数组 的 下 标 〈 键 ) ， 
当 指明 是 几 号 队员 时 就 找到 了 这 名 队员 〈 值 ) 。 

PHP 数组 比 许多 其 他 高 级 语言 中 的 数组 更 加 灵活 ， 不 但 支持 数字 索引 数组 ， 而 且 支持 以 字符 串 或 字符 
串 、 数 字 混 合 为 键 名 的 关联 数组 。 而 在 其 他 高 级 语言 ， 如 Java 或 者 C++ 等 语言 中 ， 只 支持 数字 索引 数组 。 
PHP 数组 结构 与 其 他 语言 数组 结构 的 比较 如 图 5.2 和 图 5.3 所 示 。 


HON WE 
123456 PHP 程序 员 


图 5.3 PHP 的 数组 结构 
5.2 声明 数组 


锋 1 视频 讲解 :光盘 \TM\Video\ 第 5 章 \ 声 明 数 组 .exe 
PHP 中 声明 数组 的 方式 主要 有 两 种 ， 一 种 是 应 用 数组 函数 声明 数组 ， 另 一 种 是 通过 为 数组 元 素 赋值 的 
方式 声明 数组 。 


5.2.1 数组 命名 规则 


PHP 中 声明 数组 的 规则 : 数组 的 名 称 由 一 个 “$” 【美元 符号 ) 开始 ， 第 一 个 字符 是 字母 或 下 划 线 ， 其 


@ 
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后 是 任意 数量 的 字母 、 数 字 或 下 划 线 。 例 如 : 
$array_name=array('PHP'=>'php','ASP'=>'asp',JAVA'=>'java'); // 以 字符 串 作为 数组 索引 ， 指 定 关键 字 


$_aa=array('PHP",'Java',C#,Vb'); /以 数字 作为 数组 索引 ， 从 0 开始 ， 没 有 指定 关键 字 
$array[]=value1; 

上 述 都 是 符合 命名 规则 的 数组 ， 而 下 面 的 这 两 个 数组 则 不 符合 命名 规则 。 
//$1="array(1=>'a',2=>'b',3=>'c',4=>'d")"; /不 可 以 以 数字 开头 
$@=array(PHP'=>'php'JAVA=>'java'JSP'=>jsp'):; // 不 可 以 使 用 特殊 字符 


在 同一 个 程序 中 ， 标 量变 量 和 数组 变量 都 不 能 重 名 。 例 如 ， 如 果 已 经 存在 一 个 名 称 为 $string 的 变量 ， 
而 又 创建 一 个 名 称 为 $string 的 数组 ， 那 么 前 一 个 变量 就 会 被 覆盖 。 
数组 的 名 称 是 区 分 大 小 写 的 ， 如 $String 与 $string 是 不 同 的 。 


5.2.2 通过 PHP 函数 创建 数组 


在 PHP 中 ， 可 以 创建 数组 的 函数 如 表 5.1 所 示 。 这 里 主要 讲解 通过 array0 函 数 创建 数组 。 
表 5.1 创建 数组 的 函数 


函数 功 能 
arrayO 创建 一 个 数组 (创建 数组 最 常用 的 函数 ) 
array combineO) 以 一 个 数组 为 关键 字 、 另 一 个 数组 为 值 建立 新 数组 
array fill0 用 值 填充 一 个 数组 
array padO) 用 一 个 值 把 数组 填充 到 指定 长 度 
compactO 创建 包含 变量 及 其 值 的 数组 
rangeO 创建 一 个 数组 包含 一 定 范围 内 的 元 素 


应 用 array0O 函 数 声明 数组 的 方式 如 下 : 

array array ( [mixed ...]) 

参数 mixed 的 语法 为 “key => value”， 多 个 参数 mixed 用 逗号 分 开 ， 分 别 定义 索引 和 值 。 索 引 可 以 是 
字符 串 或 数字 。 如 果 省 略 了 索引 ， 会 自动 产生 从 0 开始 的 整数 索引 。 如 果 索 引 是 整数 ， 则 下 一 个 产生 的 索 
引 将 是 目前 最 大 的 整数 索引 +1。 如 果 定 义 了 两 个 完全 一 样 的 索引 ， 则 后 面 一 个 会 覆盖 前 一 个 。 数 组 中 各 数 
据 元 素 的 数据 类 型 可 以 不 同 ， 也 可 以 是 数组 类 型 。 当 mixed 是 数组 类 型 时 ， 就 是 二 维 数组 。 

应 用 array0 函 数 声明 数组 时 , 数组 下 标 既 可 以 是 数值 索引 也 可 以 是 关联 索引 。 下 标 与 数组 元 素 值 之 间 用 
“=>” 进 行 连接 ， 不 同 数组 元 素 之 间 用 逗号 进行 分 隔 。 

应 用 array0 函 数 定义 数组 比较 灵活 ， 可 以 在 函数 体 中 只 给 出 数组 元 素 值 ， 而 不 必 给 出 键 值 。 

例 5.1 通过 array0 函 数 声明 数组 ， 并 输出 数组 中 的 值 。 《实例 位 置 : 光盘 \TM\Instance\05\5.1) 


代码 如 下 : 

<?php 

$array_name=array(PHP'=>'php',,ASP'=>jasp'"C##=>'c#); /以 字符 串 作 为 数组 索引 ， 指 定 关键 字 
print_r($array_name); /| 输出 数组 

echo "<br>"; /换行 

echo $array_name[PHP]; // 输 出 数组 中 的 索引 为 PHP 的 元 素 
echo "<br>"; 

$_aa=array('PHP",Asp','C#,Vb'); /以 数字 作为 数组 索引 ， 从 0 开始 ,没有 指定 关键 字 
print_r($_aa); // 输 出 整个 数组 

echo "<br>"; 

echo $_aa[1]; // 输 出 数组 中 的 第 一 个 元 素 

Ee 


PHP+MySQL 开发 实战 


运行 结果 如 图 5.4 所 示 。 
Py 
sw array0 来 创建 
空 数组 ， 然 后 通过 使 用 方 括号 ([] ) 来 添加 值 。 


Array ( [PHF] => zhp [8 


je ( [0] =» Per 
二 


PHP 提供 创建 数组 的 array0 语 言 结构 . 在 使 用 其 中 的 数据 时 ， 
可 以 直接 应 用 它们 在 数组 中 的 排列 顺序 取 值 ， 该 顺序 称 为 数组 的 
下 标 。 例 如 : 
<?php 
echo $_aa[ 0]; // 输 出 数组 元 素 的 第 0 个 下 标 值 


图 5.4 通过 array0 函 数 声明 数组 


?> 


结果 为 : PHP。 


和 应 用 这 种 方式 定义 数组 时 ， 下 标 默认 是 从 0 开始 的 ， 而 不 是 1， 然 后 依次 增加 1。 所 以 为 0 的 
元 素 是 指数 组 的 第 1 个 元 素 。 


5.2.3 ”通过 数组 标识 符 “[] ”创建 数 组 


PHP 中 另 一 种 比较 灵活 的 数组 声明 方式 是 通过 数组 标识 符 “[] ”直接 为 数组 元 素 赋值 。 如 果 在 创建 数组 
时 不 知 所 创建 数组 的 大 小 ， 或 在 实际 编写 程序 时 数组 的 大 小 可 能 发 生 改 变 ， 采 用 这 种 数组 创建 的 方法 较 好 。 
例 5.2 通过 数组 标识 符 “[]” 创 建 数 组 。 (实例 位 置 光盘 \TM\Instance\05\5.2) 
代码 如 下 : 
<?php 
$array[0]="PHP"; 
" 编 "; 


print 人 // 输 出 所 创建 数组 的 结构 


?> 


结果 为 : Array([0]=> PHP [1] => 编 2]=> 程 3]=> 词 [和 ]=> 典 )。 


er 


$.3 ”数组 的 类 型 


名 g 视频 讲解 :光盘 \TM\Video\ 第 5 章 \ 数 组 的 类 型 .exe 
PHP 支持 两 种 数组 : 数字 索引 数组 〈indexed array) 和 关联 数组 (associative array) ， 数 字 索 引 数 组 使 
用 数字 作为 键 ， 关 联 数组 使 用 字符 串 作为 键 。 


® 
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5.3.1 数字 索引 数组 


PHP 数字 索引 由 数字 组 成 ， 下 标 从 0 开始 ， 数 字 索 引 一 般 表示 数组 元 素 在 数组 中 的 位 置 。 数 字 索 引 数 
组 默认 索引 值 从 数字 0 开始 ， 不 需要 特别 指定 ，PHP 会 自动 为 索引 数组 的 键 名 赋 一 个 整数 值 ， 然 后 从 这 个 
值 开始 自动 增加 ， 当 然 ， 也 可 以 指定 从 某 个 位 置 开 始 保存 数据 。 

数字 索引 数组 可 以 构造 成 一 系列 键 - 值 (key-value) 对 , 其 中 每 一 对 都 是 数组 的 一 个 项 目 或 元 素 (element)。 
对 于 列表 中 的 每 个 项 目 ， 都 有 一 个 与 之 关联 的 键 (key) (或 索引 (index) ) ， 如 图 5.5 所 示 。 


值 (value) 一 一 > 值 值 本 
图 5.5 数字 索引 数组 


5.3.2 ”关联 数组 


关联 数组 的 键 名 可 以 是 数值 和 字符 串 混 合 的 形式 ， 而 不 像 数 字 索 引 数组 的 键 名 只 能 为 数字 ， 在 一 个 数 
组 中 ， 只 要 键 名 中 有 一 个 不 是 数字 ， 那 么 这 个 数组 就 叫做 关联 数组 。 

关联 数组 使 用 字符 串 索 引 (或 键 ) 来 访问 存储 在 数组 中 的 值 ， 如 图 5.6 所 示 。 关 联 索 引 的 数组 对 于 数据 
库 层 交互 非常 有 用 。 


值 (value) 一 一 一 一 > 
键 (key) 


图 5.6 关联 数组 
例 5.3 创建 一 个 关联 数组 。( 实 例 位 置 ， 光盘 \TMNInstance\0S\S.3) 
代码 如 下 : 
<?php 
$array = array("first"=>"PHP","second"=>"ASP","third"=>"WEB"); 
echo $array["second"]; /| 输出 索引 为 second 的 元 素 的 值 
S$array["third"]=" JAVA"; /为 索引 为 third 的 元 素 重 新 赋值 
echo $array["third"]; /| 输出 索引 为 third 的 元 素 的 值 
?> 


结果 为 : ASP JAVA。 


ci 

[ES 关联 数组 的 键 名 可 以 是 任何 一 个 整数 或 字符 囊 。 如 果 键 名 是 一 个 字符 囊 ， 不 要 忘记 给 键 名 或 
索引 加 上 定 界 修饰 符 : 单 引号 (') 或 双 引 号 (")。 对 于 数字 索引 数组 ， 为 了 避免 不 必要 的 麻烦 ， 最 好 也 
加 上 定 界 符 。 


5.4 输出 数组 


名 4 视频 讲解 : 光盘 \TM\Video\ 第 5 章 \ 输 出 数组 .exe 
PHP 中 对 数组 元 素 进行 输出 可 以 通过 输出 语句 来 实现 ， 如 echo、print 语句 等 ， 但 应 用 这 种 输出 方式 只 


全 
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能 对 某 数组 中 某 一 元 素 进行 输出 。 而 通过 print r0 和 var_dump0 函 数 可 以 将 数组 结构 进行 输出 。 

print_r0 函 数 的 语法 如 下 : 

bool print_r ( mixed expression ) 

如 果 该 函数 的 参数 expression 为 普通 的 整 型 、 字 符 型 或 实 型 变量 ， 则 输出 该 变量 本 身 ， 如 果 该 参数 为 数 
组 ， 则 按键 值 和 元 素 的 顺序 显示 出 该 数组 中 的 所 有 元 素 。 

例 5.4 使 用 print_ r0 函 数 输出 数组 的 结构 。〔 实 例 位 置 ， 光盘 \TM\Instance\05\5.4) 

代码 如 下 : 

<?php 

$array=array(1=>"PHP 编程 词典 ",2=>"C# 编 程 词典 ",3=>"JAVA 编程 词典 "); 


print_r($array); 
> 


结果 为 ，Array ( [1] => PHP 编程 词典 [2] => C# 编 程 词典 [3] => JAVA 编程 词典 )。 

var_dump0 函 数 可 以 输出 数组 (或 对 象 )、 元 素数 量 以 及 每 个 字符 串 的 长 度 ， 还 能 够 以 缩 进 方式 输出 数 
组 或 对 象 的 结构 。 

语法 : 

void var_dump( mixed expression [,mixed expression [,...]]) 

例 5.5 通过 var dump0) 函 数 输 出 数组 的 结构 。〔 实 例 位 置 ， 光盘 \TM\Instance\05\5.5) 

代码 如 下 : 

<?php 

$array=array("PHP 开发 实战 宝典 ","PHP 从 入 门 到 精通 "," 学 通 PHP 的 24 堂 课 "); 

var_dump($array); 

$arrays=array('first=>" PHP 开发 实战 宝典 ",'second'=>"PHP 从 入 门 到 精通 ",'third'=>" 学 通 PHP 的 24 堂 课 "); 

var_dump($arrays); 

?> 


运行 结果 如 图 5.7 所 示 。 


图 5.7 通过 var dump0 函 数 输出 数组 的 结构 
5.5 数组 的 构造 
铭 i 视频 讲解 : 光盘 \TM\Video\ 第 5 章 \ 数 组 的 构造 .exe 


5.5.1 创建 一 维 数组 


当 一 个 数组 的 元 素 是 变量 时 ， 称 这 个 数组 为 一 维 数组 。 一 维 数组 是 最 普通 的 数组 ， 它 只 保存 一 列 内 容 。 
例 5.6 创建 两 个 一 维 数组 , 一 个 是 数字 索引 数组 , 另 一 个 是 关联 数组 。( 实 例 位 置 : 光盘 \TM\Instance\05\5.6) 


代码 如 下 : 
<?php 
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$array=array("PHP 开发 实战 宝典 ","PHP 从 入 门 到 精通 ","PHP 求职 宝典 "); 

print_r($array); 

$arrays=array('first'=>" PHP 开发 实战 宝典 ",'second'=>"PHP 从 入 门 到 精通 ",'third'=>" PHP 求职 宝典 "); 
print_r($arrays); 

?> 


行 结果 如 图 5.8 所 示 。 
5.5.2 ”创建 二 维 数组 | 
| re ey ER 
一 个 数组 的 元 素 如 果 是 一 维 数组 ， 则 称 这 个 数组 是 二 下 CTT E 丰 ELITE 
维 数组 。 二 维 数 组 的 定义 和 使 用 与 一 维 数组 基本 相同 ， 唯 图 5.8 创建 并 输出 一 维 数组 


一 的 区 别 是 二 维 数组 的 元 素 仍然 是 数组 

例 5.7 声明 一 个 二 维 数组 。《〈 实 例 位 置 : 光盘 \TM\Instance\05\5.7) 

代码 如 下 : 

<?php 

Sstr = array ( 

"PHP 类 图 书 "=>array ("PHP 求职 宝典 ","PHP 经 典 编程 ","PHP 开发 实战 宝典 ")， 
"JAVA 类 图 书 "=>array ("a"=>"JAVA 范例 手册 ","b"=>"JAVA WEB 范例 宝典 ")， 
"ASP 类 图 书 "=>array ("ASP 学 习 手 册 ",2=>"ASP 范例 宝典 ","ASP 开发 实战 宝典 ") 


); /声明 数组 
print_r ( $str) ; /| 输出 数组 元 素 
i 
结果 为 
Array( 


[PHP 类 图 书 ] => Array( 
[0] => PHP 求职 宝典 
[1] => PHP 经 典 编程 
[2] => PHP 开发 实战 宝典 


) 
[JAVA 类 图 书 ] => Array( 
[a] => JAVA 范例 手册 
[b] => JAVA WEB 范例 宝典 


[ASP 类 图 书 ] => Array( 
[0] => ASP 学 习 手册 
[2] => ASP 范例 宝典 
[3] => ASP 开发 实战 宝典 
和 
着 
上 面 的 代码 实现 一 个 二 维 数组 的 声明 ， 按 照 同 样 的 思路 可 以 创建 更 高 维 数 的 数组 ， 如 三 维 数组 或 四 维 


5.6 遍历 数组 


本 | 视频 讲解 : 光盘 \TM\Video\ 第 5 章 \ 遍 历数 组 .exe 
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遍历 数组 中 的 所 有 元 素 是 常用 的 一 种 操作 ， 在 遍历 的 过 程 中 可 以 完成 查询 或 其 他 功能 。 如 去 商场 购物 ， 
如 果 想 要 买 一 台电 脑 ， 就 需要 在 商场 中 得 一 遍 ， 看 是 否 有 满意 的 机 型 ， 而 得 商场 的 过 程 就 相当 于 遍历 数组 
的 操作 。 在 PHP 中 遍历 数组 的 方法 有 多 种 。 下 面 介绍 最 常用 的 3 种 方法 。 


5.6.1 foreach 结构 遍历 数组 


遍历 数组 元 素 最 常用 的 方法 是 使 用 foreach 结构 。 foreach 结构 并 非 操作 数组 本 身 , 而 是 操作 数组 的 一 个 备份 。 
例 5.8 通过 foreach 结构 遍历 数组 中 的 数据 。〔 实 例 位 置 ， 光盘 \TM\Instance\05\5.8) 
代码 如 下 : 
<?php 
$name=array( 编程 词典 网 ', 编程 体验 网 ', 编程 资源 网 ); /声明 数组 
Surl = array('0'=>'Wwww.mrbccd.com','1'=>'www.mingribook.com','2'=>'Www. oo .com',); /声明 数组 
foreach ($name as $key=>$value){ // 遍 历数 组 
echo $value." "$url[$key].'<br><br>"; // 输 出 数组 中 的 数据 
} 
?> 


运行 结果 如 图 5.9 所 示 。 


5.6.2 list() 函 数 遍 历数 组 


freach 结 机 遍历 数组 


list0 函 数 把 数组 中 的 值 赋 给 一 些 变 量 .与 aray0 函 数 类 似 ， > 
list0 函 数 不 是 真正 的 函数 ， 而 是 一 种 语言 结构 。list0 函 数 仅 能 i 
用 于 数字 索引 且 索 引 值 从 0 开始 的 数组 ， 其 语法 格式 如 下 : 
void list ( mixed ...) 
参数 mixed 为 被 赋值 的 变量 名 称 。 
例 5.9 应 用 each0 和 list0 函 数 ， 输 出 存储 在 数组 中 的 用 5.9 通过 foreach 结构 遍历 数组 中 的 数据 
户 登录 信息 。〔 实 例 位 置 ， 光盘 \TM\Instance\05\5.9) 
具体 开发 步骤 如 下 。 
(1) 应 用 开发 工具 (如 Dreamweaver) ， 新 建 一 个 PHP 动态 页 ， 命 名 为 index.php。 
(2) 应 用 HTML 标记 设计 页 面 ， 首 先 建立 用 户 登 录 表 单 ， 用 于 提交 用 户 登 录 信息 ， 然 后 应 用 each0 函 数 所 
取 全 局 数组 $ POST 中 的 内 容 ， 并 最 终 应 用 while 循环 输出 用 户 所 提交 的 注册 信息 。 代 码 如 下 : 
< 一 一 一 一 一 定义 用 户 登录 表单 一 一 一 一 一 一 
<form name="form1" method="post"> 
<table width="233" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="62” height="24"” align="right” bgcolor="#FFFFFF"><span class="STYLE3"> 用 户 名 : 
</span></td> 
<td width="189" height="24" bgcolor="#FFFFFF"><input name="username" type="text” id="user" 
size= "18"></td> 
</tr> 
<tr> 
<td height="24"” align="right” bgcolor="#FFFFFF"><span class="STYLE3"> 密 &nbsp:&nbsp; 码 : 
</span></td> 
<td height="24" bgcolor="#FFFFFF"><input name="pwd" type="password" id="pwd" size="18"></td> 
</tr> 
<tr align="center" bgcolor="#CCFF33"> 
<td height="24" colspan="2" bgcolor="#FFFFFF"> 
<input type="image" name="imageField" src="images/bg1.JPG">&nbsp;&nbsp; 
<input type="image” name="imageField2" src="images/bg2.JPG" onClick="form.reset();return 


false;"></td> 
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</tr> 
</table> 
</form> 
<?php 
while(list($name, $value)=each($_POST)X // 输 出 用 户 登录 信息 
if($name!="imageField_x" and $name!="imageField_y" X{ 
echo "$name=$value<br>"; 
} 
} 
?> 


G3) 在 正 浏 览 器 中 输入 地 址 , 按 Enter 键 , 输入 用 户 名 及 密码 ,， 单 击 “ 提 交 ” 按 钮 。 运行 结果 如 图 5.10 
所 示 。 


/ 
B 
BD 明 each(O) 函 数 用 于 返回 当前 指针 位 置 的 数组 值 ， 并 将 指针 推进 一 个 位 置 . 返回 的 数组 包含 4 个 键 ， 
键 0 和 key 包含 键 名 ， 而 键 1 和 value 包含 相应 的 数据 。 如果 程序 在 执行 each0 函 数 时 ， 指 针 已 经 位 于 数 
组 末尾 ， 则 返回 false。 


5.6.3 for 语句 遍历 数组 


想 要 通过 for 循环 遍历 数组 ,首先 必须 应 用 countO 函 数 获取 数组 中 的 单元 数目 ， 然 后 将 数组 中 的 单元 数 
目 作 为 for 循环 的 条 件 ， 执 行 循 环 输出 。 

count0 函 数 的 语法 如 下 : 

int count ( mixed array [, int mode]) 


参数 array 为 指定 输入 的 数组 ;参数 mode 为 可 选 参数 ， 若 设 为 COUNT_RECURSIVE (或 1) ， 将 递归 
地 对 数组 计数 。 这 对 计算 多 维 数组 的 所 有 单元 尤其 有 用 。 此 参数 的 默认 值 为 0。 

有 关 for 循环 语句 的 详细 讲解 可 参考 第 3 章 流程 控制 语句 的 3.3.3 节 ， 这 里 不 再 歼 述 。 

例 5.10 本 实例 应 用 count0 函 数 和 for 语句 循环 输出 数组 中 的 数据 。( 实 例 位 置 : 光盘 \TM\Instance\05\5.10) 


代码 如 下 : 

<?php 
$name=array( 编程 词典 网 ', 编程 体验 网 ', 编程 资源 网 ); /声明 数组 
Surl = array('0=>'www.mrbccd.com',1=>www. mingribook.com','2=>'www.mingrisoftcom',); /声明 数组 
for($i=0; $i<count($name);$i++) { /根据 数组 中 元 素 个 数 ， 循 环 遍历 数组 

echo $name[0]." " $url[0].'<br><br>"; /| 输出 数组 中 的 数据 

} 

?> 


运行 结果 如 图 5.11 所 示 。 


o 
人 


fo 循环 遍历 数组 


编程 间 整 网 一 一 pw. mrbcod, com 
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5.11 应 用 count0 函 数 和 for 语句 循环 输出 数组 中 的 数据 
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5.7 PHP 全 局 数组 


应 用 PHP 提供 的 全 局 数组 ， 可 以 获取 大 量 与 环境 有 关 的 信息 。 例 如 ， 可 以 应 用 这 些 数组 获取 当前 用 户 
会 话 、 用 户 操作 环境 和 本 地 操作 环境 等 信息 。 下 面 对 PHP 中 常用 的 全 局 数组 进行 介绍 。 


5.7.1 $_SERVERI 全 局 数组 


$_SERVER[] 全 局 数组 包含 由 Web 服务 器 创建 的 信息 , 应 用 该 数组 可 以 获取 服务 器 和 客户 配置 及 当前 请 
求 的 有 关 信息 。 下 面 对 $_SERVER[] 数 组 进行 介绍 ， 如 表 5.2 所 示 。 


表 5.2 $_SERVER0 全 局 数组 


数组 元 素 说 明 
$ SERVER['SERVER_ ADDR'] 当前 运行 脚本 所 在 服务 器 的 他 地 址 
当前 运行 脚本 所 在 服务 器 主机 的 名 称 。 如 果 该 脚本 运行 在 一 个 虚拟 主机 上 ， 


OO 该 名 称 由 虚拟 主机 所 设置 的 值 决定 


访问 页 面 时 的 请 求 方法 ， 如 GET、HEAD、POST、PUT。 如 果 请 求 的 方式 是 
$ SERVER[REQUEST _ METHOD'] | HEAD，PHP 脚本 将 在 送出 头 信息 后 中 止 这 意味 着 在 产生 任何 输出 后 ， 不 


再 有 输出 缓冲 ) 

$ SERVER[REMOTE ADDR'] 正在 浏览 当前 页 面 用 户 的 IP 地 址 

$ SERVER[REMOTE HOSTI 正在 浏览 当前 页 面 用 户 的 主机 名 。 反 向 域名 解析 基于 该 用 户 的 
REMOTE ADDR 

$_ SERVER[REMOTE PORTI] 用 户 连 接 到 服务 器 时 所 使 用 的 端口 


当前 执行 脚本 的 绝对 路 径 名 。 注 意 : 如 果 脚本 在 CLI 中 被 执行 ， 作 为 相对 路 
$_ SERVER['SCRIPT FILENAME'] 径 ， 如 fle.php 或 者 .Jfile php，$_ SERVER['SCRIPT FILENAME'] 将 包含 用 户 
指定 的 相对 路 径 

服务 器 所 使 用 的 端口 ， 默 认为 80。 如 果 使 用 SSL 安全 连接 ， 则 该 值 为 用 户 设 
置 的 HTTP 端口 

$_SERVER['SERVER_SIGNATURE'] | 包含 服务 器 版 本 和 虚拟 主机 名 的 字符 串 


$ SERVER[DOCUMENT ROOT] 当前 运行 脚本 所 在 的 文档 根 目录 。 在 服务 器 配置 文件 中 定义 


$_SERVER['SERVER PORT'] 


例 5.11 通过 $_SERVER[] 全 局 数组 获取 服务 器 和 客户 端的 IP 地 址 ， 客 户 端 连接 主机 的 端口 号 和 服务 
器 的 根 目录 。《〔 实 例 位 置 ， 光盘 \TM\Instance\05\5.11) 
其 代码 如 下 。 
<?php 
echo "当前 服务 器 IP 地 址 是 : <b>".$_SERVER[SERVER_ADDR]."</b><br>"; 
echo "当前 服务 器 的 主机 名 称 是 : <b>".$_SERVER[SERVER_NAME']."</b><br>"; 
echo "客户 端 IP 地 址 是 : <b>".$_SERVER[REMOTE_ADDR'."</b><br>"; 
echo "客户 端 连接 到 主机 所 使 用 的 端口 : <b>".$_SERVER[REMOTE_PORT]."</b><br>"; 
echo "当前 运行 的 脚本 所 在 文档 的 根 目录 : <b>".$_SERVER[DOCUMENT_ROOT]."</b><br>"; 
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运行 结果 如 图 5.12 所 示 。 条 而 和 各 地 时 :127991 
9 主权 时: localbest 
PP 地 是 : 127991 


5.7.2 $_GET0 和 $_POSTU0 全 局 数组 人 
PHP 中 提供 $_GET[] 和 $_POST[] 全 局 数组 ,分 别 用 来 接收 GET ”图 5.12 获取 服务 器 和 客户 端的 也 地址 
和 了 POST 方法 传递 到 当前 页 面 的 数据 。 
例 5.12 下 面 开发 一 个 实例 ， 获 取 用 户 的 登录 信息 。 分 别 通 过 GET 和 POST 方法 完成 数据 的 提交 ， 并 
且 应 用 $_GET[] 和 $_POST[D] 全 局 数组 获取 用 户 提交 的 数据 ， 从 返回 的 结果 中 体会 二 者 之 间 的 区 别 。 (实例 位 
置 光盘 \TM\Instance\05\5.12) 
具体 实现 步骤 如 下 : 
(1) 创建 index.php 文件 ， 同 时 定义 两 个 form 表单 ， 分 别 使 用 GET 和 POST 方法 提交 数据 ， 将 通过 
GET 方法 提交 的 数据 传递 到 getphp 文件 ， 将 通过 POST 方法 提交 的 数据 传递 到 postphp 文件 。 
(2) 创建 getphp 文件 ， 通 过 $_GET[] 全 局 数组 获取 GET 方法 提交 的 数据 。 运 行 结果 如 图 5.13 所 示 。 
ee < ea cE 


用户 儿 为: nint 
Er 


EE 


图 5.13 ”利用 $_GET 变量 的 输出 页 面 
有 具体 代码 如 下 : 


<?php 

if(isset($_GET['Submit]) and $_GET[Submit]==" 提 交 ")f 
echo "用 户 名 为 : ".$_GET[textfield]."<br>"; 
echo "密码 为 : "$_GET[textfield21; 

} 

?> 


03) 创建 post.php 文件 , 通过 $_POST[] 全 局 数组 获取 POST 方法 提交 的 数据 。 运行 结果 如 图 5.14 所 示 。 


图 5.14 利用 $_POST 变量 的 输出 页 面 


基体 代码 如 下 : 

<?php 

ifisset($_POST[Submit2]) and $_POST[Submit2]==" 提 交 "){ 
echo "用 户 名 为 : ".$_POST[textfield31]."<br>"; 
echo "密码 为 : "$_POST[textfield224]; 

) 


?> 


5.7.3 $_COOKIE 全 局 数组 


$_COOKIE[] 全 局 数组 存储 通过 http Cookie 传递 到 脚本 的 信息 。PHP 中 可 以 通过 setcookie0 函 数 设 置 


0) 
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Cookie 的 值 ， 用 $_COOKIE[] 数 组 接收 Cookie 的 值 ，$_COOKIE[] 数 组 的 下 标 为 Cookie 的 名 称 。 
例如 ， 通 过 setCookie0 函 数 创建 一 个 Cookie， 并 通过 $_COOKIE[] 全 局 数组 获取 Cookie 的 值 。 其 代码 


如 下 : 
<?php 
setCookie("mingri",' 明 日 科技 '); 
setCookie("mingri", ' 明 日 科技 time()+60); /设置 Cookie 有 效 时 间 为 60 秒 
echo " 读 取 Cookie: "$_COOKIE[mingri]; /通过 $_COOKIE0 读 取 Cookie 的 值 


?> 


5.7.4 $_ENV[] 全 局 数组 


$_ENV[] 全 局 数组 用 于 提供 与 服务 器 有 关 的 信息 。 例 如 ， 
回 $_ENV["HOSTNAME"]: 获取 服务 器 名 称 。 
$_ENV["SHELL"]: 获取 系统 shell。 


5.7.5 $_REQUEST[] 全 局 数组 


可 以 用 $_REQUEST[] 全 局 数组 获取 GET 方法 、POST 方法 和 http Cookie 传递 到 脚本 的 信息 。 如 果 在 编 
写 程序 时 ， 不 能 确定 是 通过 什么 方法 提交 的 数据 ， 那 么 就 可 以 通过 $_REQUEST[] 全 局 数组 获取 提交 到 当前 
页 面 的 数据 。 


5.7.6 $_SESSION[ 全 局 数组 


$_SESSION[] 全 局 数组 用 于 获取 会 话 变量 的 相关 信息 。 
例如 , 初始 化 SESSION 变量 , 通过 $_ SESSION[] 全 局 数组 为 SESSION 变量 赋值 , 最 后 通过 $_ SESSION[] 


全 局 数组 输出 SESSION 的 值 。 其 代码 如 下 : 
<?php 


session_start(); /初始 化 SESSION 变量 
$_SESSION[mr]="mingri"; /为 SESSION 变量 赋值 
echo $_SESSION[mr] /| 输出 SESSION 变量 的 值 
es 


5.7.7 $_FILESI[] 全 局 数组 


与 其 他 全 局 数组 不 同 ，$_FILES[] 全 局 数组 为 一 个 多 维 数 组 ， 该 数组 用 于 获取 通过 POST 方法 上 传 文件 
时 的 相关 信息 。 如 果 为 单 文件 上 传 ， 则 该 数组 为 二 维 数组 ， 如 果 为 多 文件 上 传 ， 则 该 数组 为 三 维 数组 。 下 
面 对 该 数组 的 具体 参数 取 值 进行 描述 。 
$_FILES["file"]["name"]: 从 客户 端 上 传 的 文件 名 称 。 
$_FILES["file"]["type"]: 从 客户 端 上 传 的 文件 类 型 。 
$_FILES["userfile"]["size"]: 已 上 传 文件 的 大 小 。 
$_FILES["file"]["tmp_name"]: 文件 上 传 到 服务 器 后 ， 在 服务 器 中 的 临时 文件 名 。 
$_FILES["file"]["error"]: 返回 在 上 传 过 程 中 发 生 错误 的 错误 代号 。 


[sl 
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5.8 PHP 的 数组 函数 


下 面 介绍 一 些 在 实际 程序 开发 中 比较 常用 的 数组 函数 ， 如 果 需 要 使 用 数组 函数 实现 某 些 特殊 的 功能 ， 
可 以 参考 PHP 中 文 手册 ， 其 中 有 对 所 有 数组 函数 的 详细 介绍 ， 可 以 根据 所 要 实现 的 功能 进行 选择 、 学 习 或 
者 研究 。 


5.8.1 统计 数组 元 素 个 数 


区 视频 讲解 : 光盘 \TM\Video\ 第 5 章 \ 统 计数 组 元 素 个 数 .exe 

在 PHP 中 ， 应 用 count0 函 数 可 以 对 数组 中 的 元 素 个 数 进行 统计 ， 在 讲解 使 用 for 循环 遍历 数组 时 已 经 
应 用 到 ， 下 面 详细 介绍 一 下 该 函数 。 语 法 格式 如 下 : 

int count ( mixed var [, int mode]) 

参数 var 指定 操作 的 数组 对 象 ， 参 数 mode 为 可 选 参数 ， 默 认 值 是 0。 如 果 mode 的 值 设置 为 COUNT_ 
RECURSIVE (或 1) ，count0 函 数 检测 多 维 数组 。 该 函数 返回 数组 元 素 的 个 数 。 


9 
说明 如 果 countO 函 数 的 操作 对 象 是 NULL， 那 么 返回 结果 是 0。countO 函 数 对 没有 初始 化 的 变量 返 
回 0， 但 对 于 空 的 数组 也 会 返回 0。 如 果 要 判断 变量 是 否 初 始 化 ， 则 可 以 应 用 issetO 函 数 。countO 函 数 不 
能 识别 无 限 递归 。 


例 5.13 下 面 使 用 countO 函 数 统计 数组 中 元 素 个 数 , 并 输出 统计 结果 。( 实 例 位 置 : 光盘 \TM\Instance\ 
05\5.13) 


代码 如 下 : 

<?php 

$array=array(0 =>'PHP 开发 实战 宝典 ', 1 =>'JAVA 学 习 手 册 ', 2 =>'HTML 从 入 门 到 精通 '); 

echo count($array); // 统 计数 组 中 元 素 个 数 ， 并 使 用 echo 语句 输出 统计 结果 
> 


运行 结果 为 : 3。 
5.8.2 ”向 数组 中 添加 元 素 


在 PHP 中 , 使 用 array_push0 函 数 可 以 向 数组 中 添加 元 素 , 该 函数 将 传 入 的 元 素 添 加 到 某 个 数组 的 末尾 ， 
并 返回 数组 新 的 单元 总 数 。 语 法 如 下 : 

int array_push ( array array, mixed var [, mixed ...]) 

参数 array 为 指定 的 数组 ， 参 数 var 是 压 入 数组 中 的 值 。 

例 5.14 下面 使 用 array_pushO 函 数 向 数组 中 添加 元 素 ， 并 输出 添加 元 素 后 的 数组 。 (实例 位 置 : 光盘 \ 
TM\Instance\05\5.14) 

代码 如 下 : 

<?php 

$array=array(0 =>'PHP 求职 宝典 ', 1 =>'JAVA 范例 宝典 '); /声明 数组 

echo "添加 前 的 数组 元 素 :"; 

print_r($array); 

echo "<br>"; 
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array_push($array,VB 标准 教程 ,VC 从 入 门 到 精通 '); // 向 数组 中 添加 元 素 
echo "添加 后 的 数组 元 素 :"; 

print_r($array); // 输 出 添加 后 的 数组 结构 
?> 

运行 结果 如 图 5.15 所 示 。 


5.8.3 ”获取 数组 中 最 后 一 个 元 素 


| PET 


在 PHP 中 ， 通 过 array_pop0 函 数 可 以 获取 并 = 
返回 数组 中 的 最 后 一 个 元 素 ， 并 将 数组 的 长 度 减 5.15 ”使 用 array_push0 函 数 向 数组 中 添加 元 素 
1， 如 果 数 组 为 空 〈 或 者 不 是 数组 ) 将 返回 NULL。 语 法 格式 如 下 : 

mixed array_pop ( array array) 

参数 array 为 输入 的 数组 。 

例 5.15 首先 应 用 array_push0 函 数 向 数组 中 添加 元 素 ， 然 后 应 用 array_pop0 函 数 获取 数组 中 最 后 一 个 
元 素 ， 最 后 输出 最 后 一 个 元 素 值 。〈 实 例 位 置 : 光盘 \TM\Instance\05\5.15) 


代码 如 下 : 

<?php 

$array=array(0 =>'PHP 从 入 门 到 精通 ', 1 =>'JAVA 从 入 门 到 精通 '); /声明 数组 
array_push($array,'VB 开发 实战 宝典 ','VC 开发 实战 宝典 '); // 向 数组 中 添加 元 素 
$last_array=array_pop($array); // 获 取 数 组 中 最 后 一 个 元 素 
echo $last_array; /| 输 出 最 后 一 个 元 素 值 

?> 


运行 结果 为 : VC 开发 实战 宝典 。 
5.8.4 ”删除 数组 中 重复 元 素 


array_unique0) 函 数 将 数组 元 素 的 值 作为 字符 串 排序 , 然后 对 每 个 值 只 保留 第 一 个 键 名 , 忽略 所 有 后 面 的 
键 名 ， 即 删除 数组 中 重复 的 元 素 。 

语法 如 下 : 

array array_unique ( array array) 

参数 array 为 输入 的 数组 。 


Du amay_unique0 画 数 只 保留 重复 值 的 第 一 个 键 名 。 但是， 该 刍 名 并 不 是 在 未 排序 的 数组 中 
同一 个 值 的 第 一 个 出 现 的 键 名 ， 只 有 当 两 个 字符 串 的 表达 式 完 全 相同 时 ( (string) $eleml 一 = (string) 
Selem2 )， 第 一 个 单元 才 被 保留 。 


例 5.16 首先 定义 一 个 数组 ， 然 后 应 用 array_push0) 函 数 向 数组 中 添加 元 素 ， 并 输出 数组 ， 最 后 应 用 
array_unique0) 函 数 ， 删 除数 组 中 重复 元 素 ， 并 输出 数组 。( 实 例 位 置 ， 光盘 \TM\Instance\05\5.16) 


代码 如 下 : 

<?php 

S$arr_int = array ("PHP", ".NET","ASP"); // 定 义 数组 

array_push ($arr_int, "PHP","ASP"); // 向 数组 中 添加 元 素 
print_r($arr_int); // 输 出 添加 后 的 数组 
$result=array_unique($arr_int); /删除 添加 后 数组 中 重复 的 元 素 
print_r($result); // 输 出 删除 重复 元 素 后 的 数组 
?> 
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运行 结果 如 图 5.16 所 示 。 


A 
A 涪 明 使 用 unset0 函 数 可 删除 数组 中 的 菜 个 元 素 , 例如 , 将 例 5.16 


中 $arr int 数组 的 第 2 个 元 素 删除 ， 关 键 代 码 如 下 : 
unset($arr_int[1]); 5.16 ”删除 数组 中 重复 元 素 


5.8.5 ”获取 数组 中 指定 元 素 的 键 名 


获取 数组 中 指定 元 素 的 键 名 主要 是 通过 数组 函数 来 实现 , 本 节 分 别 介绍 array_search0 函 数 和 array_keys0 
函数 获取 数组 中 重复 元 素 的 所 有 键 名 。 

回 ”使 用 array_search0 函 数 可 获取 数组 中 指定 元 素 的 键 名 

使 用 array_search0 函 数 可 获取 数组 中 指定 元 素 的 键 名 ， 在 数组 中 搜索 给 定 的 值 ， 找 到 后 返回 键 名 ， 否 
则 返回 false。 其 语法 如 下 : 

mixed array_search ( mixed needle, array haystack [, bool strict]) 

array_search() 函 数 的 参数 说 明 如 表 5.3 所 示 。 


表 5.3 array_search() 函 数 的 参数 说 明 


指定 在 数组 中 搜索 的 值 ， 如 果 needle 是 字符 串 ， 则 比较 以 区 分 大 小 写 的 方式 进行 


指定 被 搜索 的 数组 
可 选 参数 ， 如 果 值 为 tue， 还 将 在 haystack 中 检查 needle 的 类 型 


/ 
NC 说 明 
atray_search() 函 数 是 区 分 字母 大 小 写 的 。 
例 5.17 下 面 使 用 amray_searchO 函 数 获 取 数组 中 元 素 的 键 名 。 实例 位 置 : 光盘 \TM\Instance\05\5.17) 


具体 代码 如 下 : 

<?php 

$arr=array(" 葡 萄 "," 山 竹 "," 橙 子 "," 西 瓜 ");  // 创 建 数组 ， 数 组 中 有 4 个 元 素 

$name=array_search(" 西 瓜 ",$arr); /使 用 array_search 获取 $arr 数组 中 “西瓜 ”的 键 名 ， 然 后 将 获取 的 结果 
赋 给 Sname 变量 

echo $name; /| 输出 结果 

?> 

运行 结果 为 : 3。 


使 用 array_keys0 函 数 获取 数组 中 重复 元 素 的 所 有 键 名 。 

如 果 查 询 的 元 素 在 数组 中 出 现 两 次 以 上 ， 那 么 array_keys0 函 数 则 返回 第 一 个 匹配 的 键 名 。 如 果 想 要 返 
回 所 有 匹配 的 键 名 ， 则 需要 使 用 array_keys(O) 函 数 。 语 法 如 下 : 

array array_keys ( array input [, mixed search_value [, bool strict]] ) 

array_keys0 返 回 input 数组 中 的 数字 或 者 字符 串 的 键 名 。 如 果 指 定 可 选 参数 search_value， 则 只 返回 该 
值 的 键 名 。 否 则 input 数组 中 的 所 有 键 名 都 会 被 返回 。 

例 5.18 下面 使 用 amray keys0O 函 数 来 获取 数组 中 重复 元 素 的 所 有 键 名 。( 实 例 位 置 :光盘 \TMNInstanceWs\s.18) 


具体 代码 如 下 : 

<?php 

$arr=array(" 葡 萄 "," 山 竹 "," 枪 子 "," 西 瓜 "," 山 竹 "); 

$name=array_keys($arr," 山 往 "); /使 用 array_keys() 函 数 获取 $arr 数组 中 “山竹 ”的 所 有 键 值 


DD) 
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print_r($name); /因为 array_keys() 函 数 返 回 的 是 数组 类 型 的 值 ， 所 以 使 用 print_r 输出 


?> 


运行 结果 为 : Array ([0] => 1 [1] => 4)。 


5.9 实 战 
区 视频 讲解 : 光盘 \TM\Video\ 第 5 章 \ 实 战 .exe 


5.9.1 获取 上 传 文件 的 数据 


例 5.19 获取 上 传 文件 的 数据 ， 需 要 使 用 $_FILES[] 全 局 数组 中 的 S_ FILES["file"]["name"] 实 现 ， 本 实例 
使 用 该 全 局 数组 获取 上 传 文件 的 名 称 。《〈 实 例 位置 : 光盘 \TM\Instance\05\5.19) 


具体 代码 如 下 : 

<?php 

if(isset($_POST[Submit]) and $_POST[Submit]==" 获 取 上 传 文件 名 "){ ” // 判 断 按 钮 的 值 是 否 存 在 ， 是 否 为 空 
echo $_FILES["file"]["name"]; // 输 出 上 传 文件 名 

出 

?> 


运行 结果 如 图 5.17 所 示 。 


5.9.2 ”投票 管理 系统 


例 5.20 在 开发 一 个 投票 管理 系统 时 , 经 常 需要 在 后 台 
添加 投票 选项 到 投票 系统 ， 作 为 投票 的 内 容 。 下 面 应 用 
explode0 函 数 对 添加 的 投票 选项 通过 “* ”进行 区 分 ， 然 后 通 
过 while 循环 语句 输出 添加 的 投票 选项 。《〈 实 例 位 置 ， 光盘 \ 


TM\Instance\05\5.20) 人 - /- 
有 具体 开发 步骤 如 下 。 ET 
(1) 应 用 开发 工具 (如 Dreamweaver) ， 新 建 一 个 PHP 图 5.17 获取 上 传 文件 的 数据 


动态 页 ， 存 储 为 index.php。 
(2) 应 用 HTML 标记 设计 页 面 , 首先 创建 投票 选项 提交 的 表单 ,然后 应 用 each0 函 数 提取 全 局 数组 $_ POST 
中 的 内 容 ， 最 后 应 用 while 循环 输出 投票 选项 内 容 。 其 关键 代码 如 下 : 


<?php 

if$_POST[Submitll="")f /判断 提交 的 值 不 为 空 
$content=$_POST[content]; /获取 提 交 的 内 容 
$data=explode("™*",$content); // 通 过 explode() 函 数 以 “*” 分 隔 提交 的 内 容 


while(list($name,$value)=each($data){ ”// 循 环 输出 数组 中 的 数据 
echo '<input type="checkbox" name="checkbox" value="checkbox">"; 
echo $value."\n"; 
. 
由 
?> 
(3) 在 正 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 输 入 投票 选项 的 内 容 ， 各 选项 间 用 “* ”进行 分 隔 ， 单 击 
“提交 ”按钮 。 运 行 结果 如 图 5.18 所 示 。 
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图 518 在 投票 管理 系统 中 应 用 explode0 函 数 
5.9.3 ”获取 用 户 注册 信息 


例 5.21 用 户 注册 是 大 多 数 网 站 都 具备 的 功能 ， 本 实例 设计 的 用 户 注册 信息 包括 用 户 名 、 密 码 、 性 别 、 
学 历 、 爱 好 、 个 人 简介 , 信息 填写 完成 后 ， 单 击 “ 提 交 ” 按 钮 ， 将 信息 输出 到 页 面 中 。 (实例 位 置 : 光盘 \IM\ 
Instance\05\5.21) 


代码 如 下 : 
<?php 
echo "用 户 名 : ".$_POST[textfield]."<br>"; // 输 出 用 户 名 
echo "密码 : ".$_POST[textfield21."<br>"; /| 输出 密码 
echo "性 别 : "$_POST[radiobutton]."<br>" /输出 性 别 
echo "学 历 : ".$_POST[select]."<br>"; /| 输出 学 历 
/下 面 为 循环 输出 爱好 信息 
echo "爱好 :"; 
$array=$_POST['checkbox'"]; 
foreach($array as $arrjf 

echo $arr."&nbsp;&nbsp"; 


} 

echo "<br>"; 

echo "个 人 简介 :".$_POST['textarea]; // 输 出 个 人 简介 
?> 


运行 效果 如 图 5.19 所 示 。 
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5.9.4 车牌 摇号 


例 5.22 通过 随机 函数 rand0 实 现 随 机 抽取 数组 中 存储 的 车 牌号 码 。 代码 如 下 : (实例 位 置 : 光盘 \IM\ 
Instance\05\5.22) 

<?php 

$numbers = array('00000', '11111", '22222', '33333', '66666", '88888"); 

Echo "车 牌号 码 :"; 

print_r($numbers); 

S$rand = rand(0,6); 

Echo "<br> 随 机 取得 车 牌号 码 :".$numbers[$rand]; 


2 


运行 效果 如 图 5.20 所 示 。 
5.9.5 ”向 数组 中 添加 元 素 


例 5.23 使 用 array_push0 函 数 可 实现 向 数组 中 添加 元 素 , 下 面 使 用 该 函数 向 $arr 数组 中 添加 两 个 元 素 。 
《实例 位 置 ， 光 盘 \TM\Instance\05\5.23) 

代码 如 下 : 

<?php 

$arr=array(" 坚 韧 "," 创 新 "); // 创 建 数组 

echo " 原 数 组 元 素 :"; 

print_r($arr); 

echo "<br>"; 

array_push($arr," 博 学 "," 笃 行 "); /向 数组 中 添加 两 个 元 素 

echo "添加 后 的 数组 元 素 :"; 

print_r($arr); /输出 添加 元 素 后 的 数组 

?> 


运行 结果 如 图 5.21 所 示 。 
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图 5.20 车 牌 摇号 图 5.21 向 数组 中 添加 元 素 


5.10 小 结 


本 章 对 数组 的 讲解 结合 了 笔者 的 实践 经 验 ， 主 要 讲解 了 数组 中 常用 的 操作 方法 ， 包 括 遍 历数 组 、 统 计 
数组 元 素 个 数 、 向 数组 中 添加 元 素 等 ， 并 且 配合 典型 的 实例 ， 加 深 读者 对 数组 函数 的 理解 。 还 对 数组 的 定 
义 、 类 型 和 构造 进行 了 介绍 。 注 意 ， 这 里 介绍 的 只 是 PHP 数组 中 的 部 分 函数 ， 有 关 PHP 数组 中 其 他 函数 的 
介绍 ， 读 者 可 以 参考 PHP 中 文 手册 。 
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5.11 学 习 成 果 检验 


1. 尝试 声明 一 个 一 维 数组 和 一 个 二 维 数组 ， 并 对 数组 元 素 进行 输出 。 (答案 位 置 ， 光 盘 \TMNInstance\ 
05\5.24) 

2. 尝试 开发 一 个 页 面 ， 应 用 explode0 函 数 以 “* ”为 分 隔 符 ， 将 字符 串 转 换 成 数组 ， 然 后 合成 一 个 新 
数组 ， 并 从 新 数组 中 随机 抽取 3 个 元 素 值 ， 最 后 输出 随机 抽取 的 元 素 ， 运 行 结果 如 图 5.22 所 示 。 (答案 位 
置 : 光盘 \TM\Instance\05\5.25) 


注 各, 每 个 这 项 辐 用 x 分 卫 符 


| 编程 词 碌 | Fphp 中 java asp 中 # 
市 Intransc | 网 P 柑 式 : 短 同 入 > 100% > 


图 5.22 添加 多 选 题 功能 


3， 尝试 开发 一 个 页 面 ， 使 用 sort0 函 数 对 指定 的 数组 进行 升序 排序 。 (答案 位 置 ， 光 盘 \TM\Instance\ 
05\5.26) 
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( 种 | 视频 讲解 .43 分钟 ) 


对 日 期 和 时 间 进 行 操作 的 程序 无 处 不 在 ,将 别 是 在 基于 Web 的 应 
用 程序 中 ， 如 表单 提交 的 有 时间、 用 户 登 录 的 时 间 、 数 据 库 中 数据 的 更 
新 和 删除 的 时 间 等 。 想 要 记录 这 些 操作 执行 的 时 刻 ， 就 需要 通过 日 期 
和 时间 来 完成 。 日 期 和 时 间 在 我 们 的 生活 中 必 不 可 少 ， 同 样 ， 在 互联 
网 中 ， 日 期 和 时 间 也 是 非常 重要 的 。 本 章 就 来 讲解 在 PHP 中 是 如 何 操 
作 日 期 和 时 间 的 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

MW 掌握 系统 的 时 区 设置 

Hi 掌握 如 何 获取 本 地 时 间 惟 

MW 掌握 如 何 获 取 当 前 日 期 和 时 间 

”党 所 如 何 格 式 化 日 期 和 时 间 

”党 所 如 何 验 证 日 期 的 有 效 性 

MW 掌握 如 何 运 用 时 间 差 
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6.1 PHP 的 时 间 概 念 


名 1 视频 讲解 : 光盘 \TMNVideo\ 第 6 章 \PHP 的 时 间 概念 .exe 
在 PHP 语言 中 , 日期、 时 间 函 数 依 赖 于 服务 器 的 地 区 设置 ， 而 PHP 默认 设置 的 是 标准 的 格林 威 治 时 间 
( 即 采 用 的 是 零 时 区 ) 。 如 果 没有 对 PHP 的 时 区 进行 设置 ， 并 且 用 户 的 当地 时 间 是 北京 时 间 ， 那 么 你 通过 
PHP 的 时 间 函 数 获取 的 时 间 就 将 比 当 地 的 北京 时 间 少 8 个 小 时 。 因此, 要 获取 本 地 当前 的 时 间 必 须 更 改 PHP 
语言 中 的 时 区 设置 。 更 改 PHP 语言 中 的 时 区 设置 有 两 种 方法 : 在 php.ini 文件 中 设置 和 通过 
date_default_timezone_set 函数 设置 。 


6.1.1 在 php.ini 文 件 中 设置 时 区 


如 4 视频 讲解 : 光盘 \TM\Video\ 第 6 章 \ 在 php.ini 文 件 中 设置 时 区 .exe 

在 php.ini 文件 中 设置 时 区 ， 需 要 定位 到 [date] 下 的 “;date.timezone =” 选 项 ， 去 掉 前 面 的 分 号 ， 并 设置 
它 的 值 为 当地 所 在 时 区 使 用 的 时 间 。 

例如 ， 如 果 当 地 所 在 时 区 为 东 作 区， 就 可 以 设置 “date.timezone =” 的 值 为 : PRC〈 中 华人 民 共 和 国 ) 、 
Asia/Hong _ Kong (香港 ) 、Asia/Shanghai (上 海 ) 或 者 Asia/Urumqi (乌鲁木齐 ) 等 ， 这 些 都 是 东 八 区 的 时 
间 ， 如 图 6.1 所 示 。 

设置 完成 后 ， 保 存 文件 ， 重 新 启动 Apache 服务 器 。 


6.1.2 ”通过 date_default_timezone_set 函数 设置 时 区 


由 于 PHP 5 对 data0 函 数 进行 了 重 写 ， 因此 ， 目 前 的 日 期 时 间 函 数 比 系统 时 间 少 8 个 小 时 。PHP 语言 
认 设 置 的 是 标准 的 格林 威 治 时 间 〈 即 采用 的 是 零 时 区 ) ， 所 以 要 获取 本 地 当前 的 时 间 必 须 更 改 PHP 语言 中 
的 时 区 设置 。 

在 应 用 程序 中 ， 日 期 、 时 间 函 数 之 前 使 用 date_default_timezone_set0 函 数 同样 可 以 完成 对 时 区 的 设置 。 
date_default_timezone_set() 函 数 的 语法 如 下 : 

date_default_timezone_set(timezone); 

参数 timezone 为 PHP 可 识别 的 时 区 名 称 , 如 果 时 区 名 称 PHP 无 法 识别 , 则 系统 采用 UTC 时 区 。 在 PHP 
手册 中 提供 了 各 时 区 名 称 列表 ， 其 中 ， 设 置 我 国 北京 时 间 可 以 使 用 的 时 区 包括 PRC〔 中 华人 民 共和 国 〉、 
Asia/Chongqing (重庆 ) 、Asia/Shanghai (上 海 ) 或 者 Asia/Urumqi (乌鲁木齐 ) ， 这 几 个 时 区 名 称 是 等 效 的 。 

设置 完成 后 ，date0) 函 数 便 可 正常 使 用 ， 不 会 再 出 现时 差 问题 。 

例 6.1 通过 date0 函 数 格式 化 输出 当前 时 刻 的 UTC 时 间 和 北京 时 间 。( 实 例 位 置 : 光盘 \TMNInstance\06\6.1) 


代码 如 下 : 

<?php 

echo "UTC 时 间 : "date("Y-m-d Hii:s"); // 显 示 默 认 的 UTC 时 间 
date_default_timezone_set("PRC" /使 用 中 华人 民 共 和 国 的 时 区 
echo "北京 时 间 : ".date("Y-m-d H:i-s"); /输出 北京 时 间 

echo "当前 时 区 : ".date_default_timezone_get(); /获取 当前 时 区 

?> 

运行 结果 如 图 6.2 所 示 。 
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ND ET 


[Date] 
Defines the defsult timszons used by the dats functions 吕 


0 


图 6.1 设置 PHP 的 时 区 图 6.2 以 不 同 的 时 区 输出 当前 时 间 
俐 
E 技巧 如 果 服 务 器 使 用 的 是 零 时 区 ， 则 不 能 对 php.ini 文件 直接 进行 修改 ， 只 能 通过 date_default 
timezone setO) 函 数 对 时 区 进行 设置 。 


6.2 时 间 稚 
殴 il 视频 讲解 :光盘 \TM\Video\ 第 6 章 \ 时 间 玲 .exe 
6.2.1 什么 是 时 间 戳 


时 间 玲 是 文件 属性 中 的 创建 、 修 改 和 访问 时 间 。 数 字 时 间 戳 服务 (Digital Time Stamp Service，DTS) 
是 Web 网 站 安全 服务 项 目 之 一 ， 能 提供 电子 文件 的 日 期 和 时 间 信 息 的 安全 保护 。 

时 间 惟 是 一 个 经 加 密 后 形成 的 凭证 文档 ， 它 包括 以 下 3 个 部 分 : 

(1) 需要 添加 时 间 戳 的 文件 用 Hash 编码 加 密 形成 摘要 。 
(2) DTS 接收 文件 的 日 期 和 时 间 信息 。 
(3) 对 接收 的 DTS 文件 加 密 。 

数字 时 间 是 由 认证 单位 DTS 来 添加 的 ， 以 DTS 接收 到 文件 的 时 间 为 依据 。 

时 间 截 的 作用 原理 是 通过 其 他 加 密 法 将 时 间 的 数值 转换 为 加 密 的 数值 ， 时 间 变 化 后 加 密 的 数值 也 随 之 
变化 。 
时 间 截 的 优点 是 ， 变 化 的 加 密 数 值 来 防止 数值 被 窃取 后 非法 重复 利用 ， 也 就 起 到 了 加 密 的 作用 。 时 间 
被 主要 依赖 于 时 间 ， 在 约定 的 一 段 时 间 内 产生 唯一 的 一 个 数值 。 


6.2.2 UNIX 时 间 戳 


在 UNIX 系统 中 , 日 期 与 时 间 表 示 为 自 1970 年 1 月 1 日 零点 起 到 当前 时 刻 的 秒 数 , 这 种 时 间 称 为 UNIX 
时 间 截 ， 以 32 位 二 进 制 数 表示 。 其 中 ，1970 年 1 月 1 日 零点 称 为 UNIX 世纪 元 。UNIX 时 间 截 提供 了 一 种 
统一 、 简 洁 的 时 间 表 示 方 式 , 在 不 同 的 操作 系统 中 均 支 持 这 种 时 间 表 示 方 式 ,同一 时 间 在 UNIX 和 Windows 
中 均 以 相同 的 UNIX 时 间 惟 表示 , 所 以 不 需要 在 不 同 的 系统 中 进行 转换 。 同 时 ，UNIX 时 间 戳 是 一 个 时 间 差 ， 
与 时 区 没有 关系 ， 无 论 当前 PHP 中 使 用 的 是 何 种 时 区 ， 其 UNIX 时 间 戳 是 唯一 的 。 

PHP 为 UNIX 时 间 戳 的 处 理 提供 了 各 种 函数 。 到 目前 的 PHP 版 本 为 止 ， 由 于 任何 已 知 的 Windows 版 本 
以 及 其 他 一 些 系统 均 不 支持 负 的 时 间 戳 ， 因 此 在 Windows 中 无 法 表示 1970 年 1 月 1 日 之 前 的 时 间 。 目 前 
UNIX 时 间 惟 是 以 32 位 二 进 制 数 表示 的 ，32 位 二 进 制 数 值 范围 为 -2147483648 一 +2147483647， 因 此 ， 目 前 
UNIX 时 间 截 可 表示 的 最 大 时 间 为 2038 年 1 月 19 日 3 点 14 分 7 秒 ， 该 时 刻 时 间 惟 为 2147483647， 对 于 该 
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时 刻 之 后 的 时 间 ， 需 要 扩展 表示 UNIX 时 间 截 的 二 进 制 位 数 。 
6.2.3 ”获取 指定 日 期 的 时 间 戳 


PHP 中 应 用 mktime0 函 数 将 一 个 时 间 转 换 成 UNIX 的 时 间 戳 值 。 

mktimeO 函 数 根 据 给 出 的 参数 返回 UNIX 时 间 截 。 时 间 截 是 一 个 长 整数 ， 包 含 了 从 UNIX 纪元 (1970 
年 1 月 1 日 ) 到 给 定时 间 的 秒 数 。 其 参数 可 以 从 右 向 左 省 略 ， 任 何 省 略 的 参数 都 会 被 设置 成 本 地 日 期 和 时 
间 的 当前 值 。 即 如 果 不 设置 任何 参数 ， 那 么 mktime0 函 数 获取 的 将 是 本 地 的 当前 日 期 和 时 间 。 

语法 如 下 : 

int mktime(int hour, int minute, int second, int month, int day, int year, int [is_dst] ) 

mktime() 函 数 的 参数 说 明 如 表 6.1 所 示 。 


表 6.1 mktime() 函 数 的 参数 说 明 


参数 说 明 
hour 小 时 数 
minute 分 钟 数 
second 秒 数 〈 一 分 钟 之 内 ) 
month 月 份 数 
da 天 数 
eaT 年 份 数 ， 可 以 是 两 位 或 四 位 数字 ，0 一 69 对 应 于 2000 一 2069，70 一 100 对 应 于 1970 一 2000 


is_dst 参数 is_dst 在 夏令 时 可 以 被 设 为 1， 如 果 不 是 则 设 为 0， 如 果 不 确定 是 否 为 夏令 时 则 设 为 -1 默认 值 ) 


外 0 站 有 效 的 时 间 鹤 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54 到 2038 年 1 月 19 日 
03:14:07 ( 此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 ). 在 Windows 系统 中 此 范围 限制 为 从 1970 年 
1 月 1 日 到 2038 年 1 月 19 日 。 


例 6.2 应 用 mktime0 函 数 获取 系统 的 当前 日 期 时 间 ， 由 于 返回 的 是 时 间 蕉 ， 所 以 还 要 通过 date0 函 数 
对 其 进行 格式 化 后 才能 够 输出 日 期 和 时 间 。 《实例 位 置 : 光盘 \TM\Instance\06\6.2) 


代码 如 下 : 
<?php 
echo "时 间 戳 : "mktime()."<br>"; // 返 回 当前 的 时 间 戳 
echo "日 期 : ".date("Y-m-d",mktime())."<br>"; /使 用 date() 函 数 输出 格式 化 后 的 日 期 
echo "时 间 :".date("H:is"mktime()); /使 用 date() 函 数 输出 格式 化 后 的 时 间 
?> 
运行 结果 如 图 6.3 所 示 。 


6.2.4 获取 当前 时 间 戳 


PHP 通过 time0 函 数 获取 当前 的 UNIX 时 间 截 ,返回 值 为 【 Ce 
从 UNIX 纪元 (格林 威 治 时 间 1970 年 1 月 1 日 00:00:00) 到 图 63 使 用 mktime0 函 数 获取 当前 日 期 时 间 
当前 时 间 的 秒 数 。 

语法 如 下 : 


int time ( void ) 
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该 函数 没有 参数 ， 返 回 值 为 UNIX 时 间 戳 的 整数 值 。 
例 6.3 应 用 time0 函 数 获取 当前 时 间 截 , 并 将 时 间 戳 格式 化 输出 。( 实 例 位 置 : 光盘 \TM\Instance\06\6.3) 


代码 如 下 : 
<?php 
SnextWeek = time() + (7 * 24 * 60 * 60); II7 days; 24 hours; 60 mins; 60secs 
echo time()."<br>"; /当前 时 间 戳 
echo "Now: ".date("Y-m-d")."<br>"; /输出 当前 日 期 
echo "Next Week: ".date("Y-m-d",$nextWeek); // 输 出 变量 nextweek 的 日 期 
?> 


运行 结果 如 图 64 所 示 。 
6.2.5 将 英文 文本 的 日 期 时 间 描 述 解析 为 UNIX 时 间 戳 


PHP 中 应 用 strtotime0O 函 数 将 任何 英文 文本 的 日 期 时 间 解 析 为 UNIX 时 间 戳 , 其 值 为 相对 于 now 参数 给 
出 的 时 间 ， 如 果 没 有 提供 此 参数 则 用 系统 当前 时 间 。 

语法 如 下 : 

int strtotime ( string time [, int now] ) 

该 函数 有 两 个 参数 。 如 果 参 数 time 的 格式 是 绝对 时 间 ， 则 now 参数 不 起 作用 ; 如果 参数 time 的 格式 是 
相对 时 间 ， 其 对 应 的 时 间 就 是 参数 now 来 提供 的 ， 当 没有 提供 参数 now 时 ， 对 应 的 时 间 就 为 当前 时 间 。 如 
果 解 析 失 败 ， 则 返回 false。 在 PHP 5.1.0 之 前 的 版 本 中 ， 本 函数 在 失败 时 返回 -1。 

例 6.4 应 用 strtotime0O 函 数 获 取 英 文 格式 日 期 时 间 字 符 串 的 UNIX 时 间 戳 ， 并 将 部 分 时 间 输 出 。 〈 实 
例 位 置 ， 光 盘 \TMNInstance\06\6.4) 


代码 如 下 : 

<?php 
echo strtotime ("now"), "\n"; /当前 时 间 的 时 间 戳 
echo "输出 时 间 :".date("Y-m-d H:i:s",strtotime ("now")),"<br>"; /1/ 输 出 当前 时 间 
echo strtotime ("10 November 2012"), "\n"; /| 输出 指定 日 期 的 时 间 稚 


echo "输出 时 间 :".date("Y-m-d Hii:s",strtotime ("10 November 2012")),"<br>"; ”// 输 出 指定 日 期 的 时 间 
echo strtotime ("+3 day"), "\n"; 
echo "输出 时 间 :".date("Y-m-d",strtotime ("+3 day")),"<br>"; 
echo "加 一 周 :".strtotime ("+1 week")."<br>"; 
echo "加 一 周 两 天 三 小 时 四 分 钟 : ".strtotime ("+1 week 2 days 3 hours 4 seconds")."<br>"; 
echo "下 周 四 : "strtotime ("next Thursday")."<br>"; 
echo "上 周一 : ".strtotime ("last Monday"), "\n"; 
?> 


运行 结果 如 图 6.5 所 示 。 


ss 2013-0L-18 站 
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图 6.4 使 用 time0 函 数 获取 当 前 时 间 戳 图 6.5 使 用 strtotime0O 函 数 获取 当前 时 间 戳 
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6.3 PHP 日 期 和 时 间 的 处 理 


区 il 视频 讲解 :光盘 \TM\Video\ 第 6 章 \PHP 日 期 和 时 间 的 处 理 .exe 
日 期 和 时 间 的 处 理 可 以 分 为 格式 化 日 期 和 时 间 、 获 取 日 期 和 时 间 信息 、 获 取 本 地 化 的 日 期 和 时 间 及 检 
验 日 期 和 时 间 的 有 效 性 等 。 


格式 化 日 期 和 时 间 
PHP 中 通过 date0 函 数 对 本 地 日 期 和 时 间 进行 格式 化 。 语 法 如 下 : 


date(string format,int timestamp) 
参数 format 指定 日 期 和 时 间 输 出 的 格式 。 有 关 参 数 format 指定 的 格式 如 表 6.2 所 示 。 


6.3.1 


参 


» 


数 


表 6.2 参数 format 的 格式 化 选项 
说 明 


小 写 的 上 午 和 下 午 值 ， 返 回 值 为 am 或 pm 


ID le | 四 | 


大 写 的 上 午 和 下 午 值 ， 返 回 值 为 AM 或 PM 

Swatch Internet 标准 时 间 ， 返 回 值 为 000 一 999 

月 份 中 的 第 几 天 ， 有 前 导 零 的 两 位 数字 ， 返 回 值 为 01 一 31 
星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 为 Mon 到 Sun 


月 份 ， 完 整 的 文本 格式 ， 返 回 值 为 January 到 December 


一 | 一 | 一 |-. | 吕 |a 


小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 
小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 0 一 23 

有 前 导 零 的 分 钟 数 ， 返 回 值 为 00 一 59 
判断 是 否 为 夏令 时 ， 如 果 是 夏令 时 返回 值 为 1， 否则 为 0 
月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 为 1~31 


星期 数 ， 完 整 的 文本 格式 ， 返 回 值 为 Sunday 到 Saturday 


判断 是 否 为 闽 年 ， 如 果 是 六 年 返回 值 为 1， 否 则 为 0 
数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 为 01 一 12 


3 个 字母 缩写 表示 的 月 份 ， 返 回 值 为 Jan 到 Dec 


数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 


与 格林 威 治 时 间 相差 的 小 时 数 ， 如 0200 


RFC 822 格式 的 日 期 ， 如 Thu，21 Dec 2000 16:01:07 +0200 
秒 数 ， 有 前 导 零 ， 返 回 值 为 00 一 59 


每 月 天 数 后 面 的 英文 后 级 ， 两 个 字符 ， 如 st、nd、rd 或 者 也 。 可 以 和 j 一 起 使 用 


指定 月 份 所 应 有 的 天 数 


本 机 所 在 的 时 区 


从 UNIX 纪元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 


星期 中 的 第 几 天 ， 数 字 表 示 ， 返 回 值 为 0 一 6 


zlz|cll- ols | lo l= lel Ic 


ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 


两 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
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续 表 
参 。 数 说 明 
Y | 4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 
z | 年 份 中 的 第 几 天 ， 返 回 值 为 0~366 
2 时 差 偏 移 量 的 秒 数 。UTC 西边 的 时 区 偏 移 量 总 是 负 的 ，UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 返 回 值 为 
-43200 一 43200 


参数 timestamp 是 可 选 的 ， 用 于 指定 时 间 戳 ， 如 果 没有 给 出 时 间 戳 则 使 用 本 地 当前 时 间 time()。 
有 关 通 过 date() 函 数 获取 系统 当前 时 间 的 方法 在 前 面 的 实例 中 已 经 应 用 过 ， 这 里 不 再 袭 述 。 
例 6.5 应 用 date0 函 数 对 日 期 进行 格式 化 ， 设 置 不 同 的 参数 ， 进 而 输出 不 同 格式 的 日 期 。 (实例 位 置 : 


光盘 \TM\Instance\06\6.5) 
代码 如 下 : 
<?php 
echo date("Y-m-d")."<br>"; 1/2012-11-10 
echo date("m.d.y")."<br>"; /11.10.12 
echo date("j, n, Y")."<br>"; /1/10, 11, 2012 
echo date("F j, Y, g:i a")."<br>"; /November 10, 2012, 5:12 am 
echo date("D M j G:i:s TY")."<br>"; liSat Nov 10 5:12:05 UTC 2012 
echo date(\Nt \ils hhve jS \d\a\y')."<br>"; ltis the 10th day 
echo date("H:is 这 是 当前 时 间 ")."<br>"; /05:12:05 这 是 当前 时 间 
echo date('h-i-s, j-m-y, 这 是 我 的 一 天 ')."<br>"; //05-12-05, 10-11-12, 这 是 我 的 一 天 
?> 


运行 结果 如 图 6.6 所 示 。 


6.6 应 用 date0 函 数 对 日 期 进行 格式 化 


ge 
为 在 PHP 语言 中 默认 设置 的 是 标准 的 格林 威 治 时 间 ， 而 不 是 北京 时 间 。 如 果 出 现 了 时 间 不 相符 的 情况 ， 
读者 可 参考 6.1.2 节 中 的 内 容 。 


6.3.2 ”获取 日 期 和 时 间 信 息 


PHP 中 通过 getdateO 函 数 获取 日 期 和 时 间 指 定 部 分 的 相关 信息 。 语 法 如 下 : 

array getdate(int timestamp) 

该 函数 返回 数组 形式 的 日 期 、 时 间 信息 ， 如 果 没有 时 间 戳 ， 则 以 当前 时 间 为 准 。 该 函数 返回 的 关联 数 
组 元 素 的 说 明 如 表 6.3 所 示 。 
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表 6.3 getdate() 函 数 返 回 的 关联 数组 元 素 说 明 


函数 说 ” 明 

seconds 秒 ， 返 回 值 为 0 一 59 

minutes 分 钟 ， 返 回 值 为 0 一 59 

hours 小 时 ， 返 回 值 为 0 一 23 

mday 月 份 中 第 几 天 ， 返 回 值 为 1 一 31 

wday 星期 中 第 几 天 ， 返 回 值 为 0〈 表 示 星 期 日 ) 一 6 (表示 星期 六 ) 
mon 月 份 数 字 ， 返 回 值 为 1 一 12 

year 4 位 数字 表示 的 完整 年 份 ， 返 回 的 值 如 2000 或 2008 
yday 一 年 中 第 几 天 ， 返 回 值 为 0 一 365 

weekda 星期 几 的 完整 文本 表示 ， 返 回 值 为 Sunday 到 Saturday 
month 月 份 的 完整 文本 表示 ， 返 回 值 为 January 到 December 
0 返回 从 UNIX 纪元 开始 的 秒 数 


例 6.6 应 用 getdate0 函 数 获取 系统 当前 的 日 期 信息 , 并 输出 返回 值 .( 实 例 位 置 : 光 盘 \TMNVInstance\06\6.6) 


代码 如 下 : 
<?php 
Sarr = getdate(); /使 用 getdate() 函 数 将 当前 信息 保存 
echo $arr[year]."-".$arr[mon]."-".$arr[mday]." "; // 返 回 当前 的 日 期 信息 
echo $arrfhours]."".$arr[minutes].":".$arr[seconds]." ".$arr[weekday]; /返回 当前 的 时 间 信 息 
echo "Today is the $arr[ydayjth of year"; /| 输出 今天 是 一 年 中 的 第 几 天 
?> 


运行 结果 如 图 6. 


6.3.3 ”获取 本 地 化 的 日 期 和 时 间 


不 同 的 国家 和 地 


7 所 示 。 


区 ， 使 用 不 同 的 时 间 、 日 期 、 货 币 的 表示 


法 ， 以 及 不 同 的 字符 集 。 例 如 ， 在 大 多 数 西方 国家 ， 都 使 用 
Friday， 但 在 以 汉语 为 主 的 国家 中 ， 都 使 用 星期 五 ， 虽 然 都 是 


同一 个 含义 ， 但 表示 
地 化 环境 。 


的 方式 却 不 尽 相 同 。 这 时 ， 就 需要 设置 本 


全 三 地 Intranee| 课 P 鲁 式 : 休 用 


在 PHP 中 应 


setlocale0 函数 设置 本 地 化 环境 ， 应 用 。 图 57 用 getdate0 函 数 获 取 时 间 信息 


strftime0O 函 数 根 据 区 域 设 置 格式 化 本 地 日 期 和 时 间 。 
1. setlocale() 函 数 


setlocale( 〇 函数 


以 改变 PHP 默认 的 本 地 化 环境 。 语 法 如 下 : 


string setlocale(string category, string locale) 


参数 category 的 


选项 如 表 6.4 所 示 。 参 数 locale 如 果 为 空 ， 就 会 应 用 系统 环境 变量 的 locate 或 LANG 的 


值 ， 否则， 就 会 应 
为 繁体 中 文 。 


locale 参数 所 指定 的 本 地 化 环境 。 如 en_US 为 美国 本 地 化 环境 ，chs 则 指 简体 中 文 ，cht 


oh. 
入 明 对 于 Windows 平台 的 用 户 ， 可 以 登录 http://msdn.microsoft.com 来 获取 语言 和 国家 (地 区 ) 的 
编码 列表 。 如 果 是 UNIX/Linux 系统 ， 则 可 以 使 用 locale-a 命令 来 确定 所 支持 的 本 地 化 环境 。 
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表 6.4 category 参数 选项 及 说 明 


参数 说 明 
LC ALL 包含 了 下 面 所 有 的 设置 本 地 化 规则 
LC COLLATE 字符 串 比 较 
LC CTYPE 字符 串 分 类 和 转换 ， 如 转换 大 小 写 
LC MONETARY 本 地 化 环境 的 货币 形式 
LC NUMERIC 本 地 化 环境 的 数值 形式 
LC TIME 本 地 化 环境 的 时 间 形 式 


2.strftime() 函 数 

strftime() 函 数 根 据 区 域 设置 来 格式 化 输出 日 期 和 时 间 。 语 法 如 下 : 

string strftime(string format, int timestamp) 

参数 format 指定 日 期 和 时 间 输 出 的 格式 .有 关 参 数 format 识别 的 转换 标记 如 表 6.5 所 示 。 参 数 timestamp 
指定 时 间 玲 ， 如 果 没 有 指定 时 间 戳 则 应 用 本 地 时 间 。 


%d 


表 6.5 参数 format 识别 的 转换 标记 
说 明 

星期 的 简写 

星期 的 全 称 

月 份 的 简写 

月 份 的 全 称 

当前 区 域 首选 的 日 期 时 间 表 达 

世纪 值 〈 年 份 除 以 100 后 取 整 ， 范 围 为 00 一 99) 

月 份 中 的 第 几 天 ， 十 进 制 数字 《范围 为 01 一 317 


%D 和 %ny%d/%y 相同 


%e 


月 份 中 的 第 几 天 ， 十 进 制 数字 ， 一 位 的 数字 前 会 加 上 一 个 空格 (范围 为 '1'~'31') 


%g 和 %G 相同 ， 但 是 没有 世纪 值 


%G 


%h 
%H, 
%I 


4 位 数 的 年 份 ， 符 合 ISO 星期 数 参 见 %V)》 。 与 %V 的 格式 和 值 相 同 ， 但 如 果 ISO 星期 数 属于 前 一 
年 或 者 后 一 年 ， 则 使 用 那 一 年 

和 %b 相同 

24 小 时 制 的 十 进 制 小 时 数 〈 范 围 为 00 一 23) 

12 小 时 制 的 十 进 制 小 时 数 〈 范 围 为 00 一 12) 


i 


年 份 中 的 第 几 天 ， 十 进 制 数 〈 范 围 为 001 一 366) 


96m 


十 进 制 月 份 〈 范 围 为 01 一 12) 


%M 


十 进 制 分 钟 数 


换行 符 


根据 给 定 的 时 间 值 为 am 或 pm， 或 者 当前 区 域 设置 中 的 相应 字符 串 


用 am 和 p.m 符号 表示 的 时 间 


24 小 时 符号 的 时 间 


十 进 制 秒 数 
制 表 符 


当前 时 间 ， 和 %H:%M:%S 相同 


星期 几 的 十 进 制 数 表达 [1.7]，1 表示 星期 一 
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续 表 
参数 说 明 
%U 本 年 的 第 几 周 ， 从 第 一 周 的 第 一 个 星期 天 作为 第 一 天 开始 
本 年 第 几 周 的 ISO 8601:1988 格式 ， 范 围 为 01 一 53， 第 一 周 是 本 年 第 一 个 至 少 还 有 4 天 的 星期 ， 星 
期 一 作为 每 周 的 第 一 天 〈 用 %G 或 者 %g 作为 指定 时 间 戳 相应 周 数 的 年 份 组 成 7 
%W 本 年 的 第 几 周 数 ， 从 第 一 周 的 第 一 个 星期 一 作为 第 一 天 的 开始 
ow 星期 中 的 第 几 天 ， 星 期 天 为 0 


%V 


x 当前 区 域 首选 的 时 间 表示 法 ， 不 包括 时 间 
%X 当前 区 域 首选 的 时 间 表示 法 ， 不 包括 日 期 
%y, 没有 世纪 数 的 十 进 制 年 份 ( 范 围 为 00~99) 


%Y 包括 世纪 数 的 十 进 制 年 份 
%Z 时 区 名 或 缩写 
%% 文字 上 的 % 字 符 


例 6.7 应 用 setlocale0 函 数 进行 区 域 化 设置 , 然后 通过 strfime0 函 数 对 本 地 时 间 进 行 格 式 化 输出 。( 实 
例 位 置 ， 光盘 \TM\Instance\06\6.7) 
代码 如 下 : 
<?php 
setlocale(LC_ALL,"en_US"); 
echo "美国 格式 : ".strftime("Today is %A"); 
echo "<p> 
setlocale(LC_ALL,"chs"); 
echo "中 文 简体 格式 :".strftime(" 今 天 是 %A"); 
echo "<p>"; 
setlocale(LC_ALL"chty); 
echo "<p>"; 


echo "繁体 中 文 格式 :".strftime(" 今 天 是 %A"); 


?> 
运行 结果 如 图 6.8 所 示 。 


VE 
后 曙 因为 本 页 面 中 的 编码 格式 为 GB2312， 所 以 最 后 繁体 中 文 显 示 的 日 期 为 乱码 ， 如 果 将 编码 格式 
改 为 big5， 繁 体 中 文 星期 将 会 正确 显示 出 来 ， 但 其 他 文字 则 变 为 乱码 。 读 者 可 以 选择 “查看 ”/“ 编 码 ” 
命令 ， 在 弹出 的 菜单 中 选择 “繁体 中 文 (big5)” 命 令 来 查看 效果 。 


6.3.4 检验 日 期 和 时 间 的 有 效 性 


一 年 有 12 个 月 ， 一 个 月 有 31 天 (或 30 天 、28 天 、29 天 ) ， 一 星期 有 7 天 …… 这 些 常识 都 是 人 尽 皆 
知 的 事 。 但 计算 机 并 不 能 自己 分 辩 数 据 的 对 与 错 ， 只 能 依靠 开发 者 提供 的 功能 去 执行 或 检查 。PHP 中 通过 
checkdate0 函 数 检验 日 期 和 时 间 的 有 效 性 。 语 法 如 下 : 

bool checkdate(int month,int day,int year) 

其 中 ，month 的 有 效 值 为 1 一 12。day 的 有 效 值 为 当月 的 最 大 天 数 ， 如 1 月 为 31 天 、2 月 为 29 天 【〈 闫 
年 ) 。year 的 有 效 值 为 1 一 32767。 

例 6.8 应 用 checkdate() 函 数 验证 数据 录入 系统 中 输入 的 日 期 是 否 合理 ， 如 果 合 理 则 返回 “数据 录入 
成 功 ”， 否 则 返回 “您 输入 的 日 期 不 合法 ! ! ”。 (实例 位 置 ， 光 盘 \TMNInstance\06\6.8) 

国 
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操作 步骤 如 下 所 示 。 
(1) 创建 index.php 页 面 ， 添 加 form 表单 ， 设 置 表单 元 素 ， 用 于 提交 商品 的 录入 信息 ， 将 表单 中 的 数 
据 提交 到 index_ok.php 文件 中 。 


(2) 创建 index_ok.php 文件 ， 通 过 $_ POST 全 局 变量 获取 表单 中 提交 的 数据 ， 应 用 checkdate0 函 数 对 
表单 中 提交 的 日 期 进行 判断 。 代 码 如 下 : 
<?php 
if($_POST[Submit]j==true and $_POST[number]==true and $_POST[name]==true and $_POST[day] 
==true X{ // 浏 断 表单 中 提交 的 数据 是 否 为 空 
if(checkdate($month,$day,$year)==trueX{ /验证 表单 中 提交 的 日 期 是 否 合理 
echo "数据 录入 成 功 !"; 
jelsef 
echo "<script> alert( 您 输入 的 日 期 不 合法 !); history.back();</script>"; 


Jelse{ 
echo "<script> alert(' 请 详细 填写 录入 数据 !); history.back();</script>"; 
} 


2 
运行 结果 如 图 6.9 所 示 。 


Hh WS SE) i IRD) WD 
i 


ENN 全 "日 " 口 


本地 Jrtranee | 吉 P 式 :禁用 


6.8 本 地 化 日 期 6.9 ”使 用 checkdate0 函 数 验证 日 期 


6.4 实 战 


医 1 视频 讲解 :光盘 \TM\Video\ 第 6 章 \ 实 战 .exe 
6.4.1 实现 倒计时 的 功能 


例 6.9 应 用 strtotimeO 函 数 精确 计算 两 个 时 间 的 差 值 ,实现 一 个 倒计时 的 功能 。( 实 例 位 置 : 光盘 \TM\ 


Instance\06\6.9) 

实例 代码 如 下 : 

<?php 
Stime1 = strtotime(date( "Y-m-d H:i:s")); 
Stime2 = strtotime("2012-11-11 00:00:00"); /12012 年 光棍 节 
Stime3 = strtotime("2013-01-01 "); /1/12013 年 元 旦 
$sub1 = ceil(($time2 - $time1) / 3600); /60*60 
$sub2 = ceil(($time3 - $time1) / 86400); /60* 60 * 24 


echo "距离 2012 年 光棍 节 还 有 <font color=red> $sub1 </font> 小 时 !" ; 


@ 
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echo "<p>"; 
echo "距离 2013 年 元 旦 还 有 <font color=red>$sub2 </font> 天 山 "; 
?> 


P94 
OO 涪 明 ceil0 函 数 的 格式 为 float ceil(float value)， 该 函数 为 取 整 函数 ， 返 回 不 小 于 参数 value 值 的 最 小 
整数 。 如 果 有 小 数 部 分 ， 则 进 一 位 。 这 里 要 注意 ， 该 函数 的 返回 类 型 为 float 型 ， 而 不 是 整 型 。 


运行 结果 如 图 6.10 所 示 。 CA 
6.4.2 ”计算 在 线 考试 用 时 和 剩余 时 间 ee 


[ra |】 
例 6.10 在线 考试 系统 中 ， 多 数 都 是 在 规定 的 时 间 内 完成 考试 ， Ee | 
如 果 超 过 规定 时 间 ， 或 者 自动 提交 答案 ， 或 者 自动 放弃 考试 。 本 实例 
将 要 讲解 的 就 是 如 何 计算 考试 的 用 时 和 剩余 时 间 。 (实例 位 置 : 光盘 \ 


TM™M\Instance\06\6.10) 
具体 操作 步骤 如 下 。 


(1) 创建 index.php 页 面 。 首 先 介绍 考试 规则 ;然后 添加 form 


表单 ,设置 “开始 考试 "按钮, 提交 到 exam.php 页 面 ; 最 后 应 用 mktime0 图 6.10 实现 倒计时 的 功能 
函数 获取 当前 日 期 的 时 间 戳 ， 并 把 时 间 戳 的 值 赋 给 session 变量 。 代 码 
如 下 : 
<?php 
session_start(); /初始 化 session 变量 
?> 


<form name="form2" method="post" action="exam.php"> 
<input type="submit" name="Submit" value=" 开 始 考 试 " /> 


<?php 
$dates=mktime(); // 获 取 当 前 时 间 的 时 间 戳 
session_register("dates")==$dates; // 将 时 间 戳 的 值 赋 给 session 变量 
?> 
</form> 


po 


(2) 创建 exam.php 文件 。 首 先 通过 <scripf> 标 记 中 的 sre 属性 ， 调 用 屏蔽 键盘 事件 和 计算 考试 时 间 的 
JS 文件 ， 然 后 在 <body> 标 记 中 ， 通 过 onkeydown0 事 件 调 用 JavaScript 自 定义 函数 ， 实 现 对 键盘 中 事件 的 屏 
项 , 接着 , 通过 <div> 标 记 输 出 考试 用 时 和 剩余 时 间 ; 最 后 , 添加 form 表单 ， 输 出 考试 试题 ， 设 置 提交 按钮 。 
代码 如 下 : 
<?php session_start(); ?> 
< 上 -应 用 <script> 标 记 中 的 src 属性 ， 调 用 屏蔽 键盘 事件 和 计算 考试 时 间 的 JS 文件 -> 
<script language=javascript src="js/screen_event.js"></script> 
<script type="text/javascript" src="js/calculation_time.js"></script> 
<body onkeydown="keydown()"> 
<table width="750" bordercolor="#FFFFFF" bgcolor="#666666"> 
<tr> 
<td width="165" height=23 ><span class="STYLE1"> 考 试 时 间 : </span></td> 
<td width="42" ><span class="STYLE1"><font color="#FF0000">20</font> 分 钟 </span></td> 


全 
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<td width="58" ><span class="STYLE1"> 计 时 </span></td> 

<td width="193" nowrap bgcolor="#EEEEEE"><div class="STYLE1" id="show_time"></div></td> 
<td width="77" ><span class="STYLE1"> 剩 余 时 间 : </span></td> 

<td width="182" ><div class="STYLE1" id="sparetime"></div></td> 


</tr> 
</table> 
<!-- 省 略 了 添加 的 form 表单 中 的 内 容 --> 
</body> 
</html> 
(3) 创建 calculation_timejjs 文件 ， 通 过 Ajax 无 刷新 技术 计算 和 输出 考试 用 时 和 剩余 时 间 。 关 键 代码 
如 下 : 
var xmlHttp = false; // 创 建 一 个 布尔 型 变量 , 用 来 检测 是 否 为 合法 的 IE 实例 
try{ // 检 测 是 否 使 用 的 是 IE 
xmlHttp = new ActiveXObject("Msxml2.XMLHTTP"); // 如 果 JavaScript 的 版 本 大 于 5 
} catch (e) { // 如 果 不 是 ， 则 使 用 老 版 本 的 ActiveX 对 象 
try{ // 如 果 使 用 的 是 IE 浏览 器 
xmlHttp = new ActiveXObject("Microsoft. XMLHTTP"); 
} catch (e2) 0 


六 zeroewst 如 果 使 用 的 是 非 IE 浏览 器 ， 则 创建 一 个 该 对 象 的 JavaScript 实例 ”ren */ 
if (IxmlHttp && typeof XMLHttpRequest != "undefined") { 
try{ 
xmlHttp = new XMLHttpRequest(); 
j}catch(e3){ xmlHttp = false;} 


timer = window.setlnterval("ShowTime()",1000); /| 每 隔 一 秒 钟 调用 一 次 ShowTime() 函 数 
/定义 ShowTime() 函 数 以 通过 xmlHttpRequest 对 象 读 取 ShowTime.php 文件 中 的 数据 
function ShowTime(X{ 
xmlHttp.open("post","showtime.php", true); /以 POST 方法 发 送 一 个 新 请 求 
xmlHttp.onreadystatechange = function(){ 
if(xmlHttp.readyState == 4){ // 如 果 服 务 器 响应 发 出 请 求 ， 则 执行 以 下 操作 
tet = xmlHttp.responseText; // 获 取 返 回 的 响应 信息 


rr **#*################# 获取 到 | 的 信息 赋予 指 定 的 DIV 标记 enon 让 
document.getElementByld("show_time").innerHTML = tet; 


} 
} 
xmlHttp.send(null); // 发 送 请 求 
} 
time = window.setInterval("sparetime()",1000); // 每 隔 一 秒 钟 调用 一 次 sparetime() 函 数 


/sr 定义 sparetime() 函 数 以 通过 xmlHttpRequest 对 象 读 取 sparetime.php 文件 中 的 数据 ************ */ 
function sparetime(){ 


xmlHttp.open("post","sparetime.php", true); /以 POST 方法 发 送 一 个 新 请 求 
xmlHttp.onreadystatechange =function(){ 
if(xmlHttp.readyState == 4X{ // 如 果 服 务 器 响应 发 出 请 求 ， 则 执行 以 下 操作 
tet = xmlHttp.responseText; // 获 取 返 回 的 响应 信息 


/* xxxxxkxxktxxaxxtxats 将 获取 到 的 信息 赋予 指定 的 DIV 标记 szxxxxxxsxxxxxxxxxxxxxx if 

document.getElementByld("sparetime").innerHTML = tet; 

ifttet=="00:00"){ // 当 剩余 时 间 为 00:00 时 
form1.submit(); /提交 form1 表单 中 的 数据 

4 
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11 
} 
xmlHttp.send(null); 
} 
(4) 在 calculation timejs 文件 中 ， 自 定义 函数 ShowTime0 调 用 showtime.php 文件 ， 获 取 考 试用 时 ; 
sparetime.php 文件 ， 获 取 考 试 的 剩余 时 间 。 


/发 送 请 求 


自 定 义 函 数 sparetimeO 调 
showtime .php 文件 中 计算 的 是 考试 用 时 。 代 码 如 下 : 
<?php 
session_start(); /初始 化 session 变量 
$dates=$_SESSION[dates]; // 获 取 session 变量 中 存储 的 考试 开始 时 间 
$dates2=mktime(); /获取 当前 时 间 的 时 间 戳 
$dates3=$dates2-$dates; /用 当前 时 间 的 时 间 戳 减 去 考试 开始 时 的 时 间 玲 
echo date("is",$dates3); /输出 考试 当前 用 时 
ys 
sparetime.php 文件 中 计算 的 是 考试 剩余 时 间 。 代 码 如 下 : 
<?php 
session_start(); // 初 始 化 session 变量 
$dates=$_SESSION[dates]; // 获 取 session 中 存储 的 考试 开始 的 时 间 戳 
$dates1=$dates+20*60; // 计 算出 考试 规定 的 用 时 ， 时 间 为 20 分 钟 
$dates2=mktime(); /获取 当前 时 间 的 时 间 戳 
$dates3=$dates1-$dates2; /I/ 用 考试 规定 的 时 间 减 去 当前 时 间 的 时 间 稚 
// 输 出 考试 的 剩余 时 间 


echo date("i:s", $dates3); 


Ss 


(5) 创建 sereen_eventjs 文件 ， 编 写 自 定义 函数 ， 屏 项 键盘 和 鼠标 事件 。 
(6) 创建 exam _result.php 文件 ， 输 出 考试 的 结果 。 

在 线 考试 的 主页 面 运行 结果 如 图 6.11 所 示 。 

(全 明日 考试 系统 1 


TEST. . 。 


考试 时 间 。20 仿 评 。 计 时 。o 2 利 作 时间，;4 3 


故 渡 是 日 


适 言 开发 而 


+ 司 过 亏欠 贞 和 了 宁 作者 用 于 时 儿 人 人 


运 局 于 在 入 


E33| 


图 6.11 计算 在 线 考试 用 时 和 剩余 时 间 
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6.4.3 ”网 页 六 钟 


例 6.11 所 谓 网 页 曾 钟 也 就 是 一 个 日 志 提 醒 功 能 ， 通 过 判断 系统 中 当前 的 时 间 与 指定 的 某 个 时 间或 者 
时 间 段 是 否 相同 ， 如 果 相 同系 统 就 给 出 一 个 对 应 的 提示 信息 ， 提 示 用 户 该 做 什么 (如 每 天 工作 记录 的 上 传 、 
每 月 经 验 技巧 的 提交 或 者 员工 生日 的 提醒 等 ) ， 都 可 以 通过 日 志 的 形式 进行 提醒 。 在 本 实例 中 ， 通 过 当前 
时 间 戳 与 11 月 10 日 时 间 戳 进行 比较 ， 如 果 相 同 则 给 出 提示 信息 。 (实例 位 置 : 光盘 \TMNInstance\06\6.11) 


代码 如 下 : 
<?php 


Stime1 = strtotime(date("Y-m-d")); 


// 当 前 的 系统 时 间 ， 获 取 月 和 天 的 时 间 戳 


Stime2 = strtotime(date("Y")."-11-09"); // 设 置 时 间 戳 
if($time1==$time2}{ /| 判断 两 个 时 间 截 是 否 相同 
echo "<script>alert(' 勿 忘 国耻 ! ');window.location.href='index.php';</script>"; /给 出 提示 信息 


jelset 
echo "今天 不 是 一 个 特殊 的 日 子 !"; 
} 


?> 
运行 结果 如 图 6.12 所 示 。 


6.4.4 “检验 日 期 和 时 间 的 有 效 性 


例 6.12 一 年 12 个 月 、 一 个 月 31 天 (或 30 天 ，2 月 28 天 ， 头 年 为 29 天 ) ， 一 星期 7 天 …… 这 些 都 
是 基本 常识 。 但 计算 机 并 不 能 自己 分 辨 数据 的 对 与 错 ， 只 是 依靠 开发 者 提供 的 功能 去 执行 或 检查 。 在 PHP 
中 通过 checkdate() 函 数 检验 日 期 和 时 间 的 有 效 性 。〔 实 例 位 置 ， 光 盘 \TM\Instance\06\6.12) 


代码 如 下 : 

<?php 
header("Content-Type:text/html;charset=utf-8"); 
S$year = 2012; 1/2012 年 
$month =6; /6 月 份 
Sone_day =30; /30 天 


if(checkdate($month,$one_day, $yean)X{ 


echo "<font style='color:black;font-family: 文 易 露 雳 体 ;font-size:25'>2012 年 6 月 份 是 30 天 </font>"; 


jelsef 
} 


echo "<font style='color:black;font-family: 文 易 露 雳 体 ;font-size:18'>2012 年 6 月 份 是 31 天 </font>"; 


?> 
运行 结果 如 图 6.13 所 示 。 
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图 6.13 ”检验 日 期 和 时 间 的 有 效 性 
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6.4.5 ”获取 指定 时 间 的 UNIX 时 间 戳 


例 6.13 ”获取 指定 时 间 的 时 间 戳 需要 使 用 strtotime0 函 数 ， 此 函数 接受 任何 一 个 英文 文本 的 日 期 时 间 并 
将 其 转换 为 时 间 惟 ,本 实例 以 10 September 2010 为 例 获 取 此 时 刻 的 时 间 戳 。( 实 例 位 置 : 光盘 \TM\Instance\ 
06\6.13) 
实现 原理 非常 简单 ， 在 strtotimeO 函 数 中 添加 指定 时 间 参 数 即 可 。 代 码 如 下 : 
<?php 
header("Content-Type:text/html;charset=utf-8"); // 设 置 编码 集 
$str="<font style='color:red;font-size:20px'>".strtotime("20 December 2012")."</font>"， /指定 获 取 时 间 枚 
echo "20 December 2012 的 时 间 稚 为 “"; 
echo $str; 
?> 
运行 结果 如 图 6.14 所 示 。 
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图 6.14 获取 指定 时 间 的 时 间 戳 


6.5 小 结 


本 章 讲解 了 PHP 中 常用 的 日 期 和 时 间 处 理 函 数 ， 并 且 通 过 实例 讲解 了 这 些 函 数 的 典型 应 用 。 首 先 介绍 
时 区 的 划分 和 系统 时 区 的 设置 ， 然 后 介绍 了 UNIX 时 间 蕉 ， 包 括 什么 是 时 间 蕉 、 如 何 获取 时 间 蕉 ， 以 及 如 
何 将 英文 文本 格式 的 日 期 时 间 解 析 为 时 间 惟 ， 接 着 ， 讲 解 了 PHP 中 日 期 和 时 间 的 处 理 方法 ， 包 括 日 期 和 时 
间 的 格式 化 、 获 取 和 验证 有 效 性 等 ， 最 后 ， 通 过 实战 和 实战 练习 讲解 了 PHP 中 日 期 和 时 间 函 数 的 应 用 。 和 希 
望 通过 本 章 的 学 习 ， 读 者 可 以 掌握 PHP 中 常用 的 日 期 和 时 间 函 数 ， 熟 悉 常 用 功能 的 实现 方法 。 


6.6 ”学习 成 果 检 验 


1. 获取 指定 任意 一 天 的 时 间 。 格 式 为 YYY-MM-DD HH:MM:SS。( 答 案 位 置 : 光盘 \TM\Instance\06\6.14) 
2. 将 时 间 截 作为 随机 数 的 种 子 ， 生 成 验证 码 。 (答案 位 置 : 光盘 \TM\Instance\06\6.15) 
3. 如 何 计 算 当 前 时 间距 离 下 班 时 间 还 有 多 少时 间 ? 〈 答 案 位 置 : 光盘 \TMNInstance\06\6.16) 


第 / 


程序 调试 与 异常 处 理 


(名 视频 讲解 : 72 分 钟 ) 


即使 是 具有 多 年 开发 经 验 的 程序 开发 人 员 ， 在 开发 项 目 时 也 难免 
会 发 生 代 码 编写 错误 ， 发 生 错误 其 实 并 不 可 怕 ， 只 要 认真 分 析 、 和 努力 
思考 就 一 定 能 够 将 问题 解决 。 调 试 既 是 一 种 技术 ， 也 是 一 种 艺术 。 随 
着 经 验 的 不 新 积 果 ， 我 们 就 会 更 熟练 地 发 现 并 且 纠 正 错 误 ， 这 其 中 的 
关键 就 是 勇敢 地 去 面 对 错 误 ， 不 要 气 馆 。 本 章 将 介绍 一 些 调 试 PHP 和 
My5QL 代码 的 实用 技巧 ， 通 过 本 章 的 学 习 定 能 使 初学 者 受益 菲 浅 、 能 
力 有 所 提高 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 


让 了 解 程序 调试 的 基本 流程 

由 了 解 PHP 中 常见 错误 类 型 

让 了 解 PHP 中 常见 语法 错误 

H 掌握 PHP 的 基本 调试 策略 

H 掌握 如 何 处 理 MySQL 数据 库 中 文 的 乱码 问题 
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7.1 程序 基本 调试 流程 


即使 是 一 名 具有 多 年 开发 经 验 的 编程 人 员 ， 在 编写 代码 时 也 不 
可 能 一 次 通过 ， 往 往 要 经 过 多 次 调试 处 理 才能 实现 某 一 功能 。 具 备 CO tt 
基本 的 调试 能 力 和 策略 ， 不 仅 可 以 拓展 开发 人 员 的 编程 思路 ， 更 能 y 
有 效 缩短 项 目的 整体 开发 周期 ， 是 开发 人 员 所 必 备 的 素质 。 在 实际 CD 分 析出 错 的 原因 CD 
应 用 中 ， 程 序 的 基本 调试 流程 如 图 7.1 所 示 。 

从 上 述 调试 流程 中 可 以 判断 ， 错 误 调试 的 过 程 是 一 个 往复 的 过 
程 ， 也 就 是 说 ， 在 对 程序 中 的 错误 进行 调试 时 ， 可 能 要 进行 多 次 修 ”CD 查找 错误 根源 
改 才 能 成 功 。 在 实际 开发 中 ， 开 发 人 员 可 以 根据 自己 的 经 验 选择 适 


合 自己 的 错误 调试 方法 。 下 面具 体 介绍 错误 调试 的 一 般 流程。 ER 
(1) 根据 错误 提示 判断 错误 原因 : 程序 错误 一 般 分 为 开发 时 错 。 记 门 或 全 用 调试 工具 修改 口 忆 
误 和 运行 时 错误 。 开 发 时 错误 一 般 是 由 于 开发 人 员 的 关键 字 书 写 错 


误 造 成 的 ， 现 在 很 多 开发 工具 都 支持 开发 时 错误 提示 ， 例 如 ， 在 开 
no 人 
开发 时 错误 。 运 行 时 错误 是 由 于 开发 人 员 代码 逻辑 编写 错误 等 原因 
造成 的 ， 一 般 可 由 编译 工具 在 代码 编译 阶段 提示 错误 ， 开 发 人 员 根 图 7.1 程序 基本 调试 流程 
据 错误 提示 就 可 以 定位 到 错误 代码 的 大 概 位 置 。 

(2) 分 析出 错 原因 : 定位 到 出 错 代码 的 大 概 位 置 后 ， 根 据 错误 提示 分 析 错误 原因 ， 如 关键 字 拼 写 错误 、 
编码 逻辑 错误 等 。 

(3) 查找 错误 根源 ， 分 析出 错 的 基本 原因 后 ， 还 需要 进一步 分 析 错误 根源 ， 不 要 认为 只 将 错误 改正 就 
可 以 了 ， 读 者 在 学 习 编写 程序 时 要 有 知 其 然 、 并 知 其 所 以 然 的 精神 ， 这 样 才能 不 断 积累 提高 。 

(4) 备份 代码 :找到 出 错 原因 后 ， 不 应 该 立即 对 代码 进行 更 改 ， 首 先 需要 备份 出 错 代码 ， 防 止 在 更 改 
过 程 中 造成 更 大 错误 。 

(5) 修改 可 疑 代码 ， 完 成 代码 备份 后 ， 开 发 人 员 可 以 对 错误 代码 进行 修改 ， 之 后 再 运行 调试 ， 也 就 是 
说 ， 如 果 代码 修改 后 再 一 次 发 生 错误 ， 那 么 再 回 到 调试 流程 的 第 1) 步 。 

上 面 介绍 的 程序 调试 流程 是 最 基本 的 调试 过 程 ， 读 者 不 必 循 规 蹊 矩 ， 可 以 在 开发 中 根据 实际 情况 总 结 
出 适合 自己 的 调试 方法 和 流程 。 


7.2 PHP 中 的 错误 类 型 


在 实际 项 目 开 发 中 ， 可 能 会 遇 到 各 种 错误 ， 不 过 只 要 开发 人 员 在 平时 学 习 和 工作 中 多 积累 、 多 总 结 ， 
就 可 以 在 开发 过 程 中 游 思 有 余 ， 轻 松 地 解决 程序 中 的 问题 。 在 项 目 开 发 中 常见 的 错误 类 型 包括 语法 错误 、 
语义 错误 、 逻 辑 错误 和 注释 错误 等 。 
7.2.1 ”语法 错误 


吉 4 视频 讲解 :光盘 \TM\Video\ 第 7 章 \ 语 法 错误 .exe 
每 一 种 计算 机 语言 都 有 自身 的 语法 格式 ， 开 发 人 员 必 须 按照 这 种 语法 规则 书写 代码 ， 否 则 会 发 生 语 法 


_ 国 ) 
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错误 。PHP 语言 自身 有 其 独特 的 语法 规则 ， 如 变量 名 前 要 加 “$”、 每 行 要 以 “;” 结 束 、 关 键 字 大 小 写 敏 感 
等 。 不 过 ，PHP 自身 是 一 种 弱 声 明 语 言 ， 也 就 是 说 PHP 中 的 变量 不 需 声 明 即 可 使 用 ， 这 就 为 开发 人 员 带 来 
了 很 大 的 方便 。PHP 中 常见 的 语法 错误 有 如 下 几 种 形式 。 
1. 缺少 结束 符 引 起 的 错误 
编写 PHP 代码 时 ， 要 求 每 一 行 以 “;” 结 束 ， 如 果 代 码 编写 人 员 因 玻 忽 未 写 结束 符 “;”， 在 运行 或 调试 
程序 时 就 会 发 生 错 误 。 
例 7.1 使 用 for 循环 输出 1 一 20 之 间 的 数字 , 并 设置 缺少 结束 符 错 误 .( 实 例 位 置 :光盘 \TM\Instance\07\7.1) 
<?php 
for($i=1; $i<21; $i++) 
{ 
echo $i // 没 写 结束 符 
} 


re 

运行 上 述 代 码 ， 将 在 页 面 中 输出 如 图 7.2 所 示 的 错误 提示 。 

分 析 页 面 中 的 错误 ， 从 语意 中 可 以 判断 程序 的 第 5 行 应 该 以 “;” 结 束 ， 而 不 应 该 以 “}” 结 束 ， 也 就 是 
说 ， 程 序 将 “}” 当 作 第 5 行 的 结束 符 ， 而 非 for 循环 体 的 结束 标志 。 

2. 缺少 单 引号 或 双 引 号 引起 的 语法 错误 

和 其 他 语言 不 同 ，PHP 中 无 论 使 用 单 引号 还 是 双 引 号 ， 所 引 部 分 都 当 作 字符 串 常 量 ， 区 别 是 使 用 单 引 
号 的 效率 较 双 引号 高 ， 而 双 引 号 字符 串 中 可 以 包含 变量 并 能 进行 区 分 。 在 编写 代码 时 ， 开 发 人 员 可 能 由 于 
书写 错误 ， 使 用 单 引号 或 者 双 引号 时 少 书写 一 个 ， 或 字符 串 两 侧 使 用 的 引号 不 一 致 而 导致 语法 错误 。 

例 7.2 使 用 echo 语句 输出 一 个 字符 串 , 并 得 到 引号 不 一 致 错误 。( 实 例 位置 : 光盘 \TMNInstance\07\7.2) 

<?php 

echo "mingri' /引号 不 一 臻 


?> 


Parst yn1aY eTror, Unerpect *,” 加 Parse error: syntax error, unexpected $end in 
in :VAppServ mm : ine5 ~ | E: \AppServ\wwe\TINOT\T. 2\indox. php on line 4 
全 市 翅 Jntranec | 需 P 杰 式 :禁用 | I 篇 地 地 lntranet | 课 p 模 式 : 委 用 和 > 


7.2 缺少 结束 符 错误 图 7.3 引号 不 一 致 错误 

从 错误 提示 中 可 以 判断 ， 导 致 错误 的 原因 是 没有 结束 标志 ， 这 说 明 在 程序 编译 时 将 字符 串 后 的 单 引号 
当 作 字符 串 的 一 部 分 ， 而 非 字符 串 的 结束 标志 。 

3. 缺少 括号 引起 的 语法 错误 

与 C/C++ 语法 类 似 , PHP 中 诸如 for 循环 、while 循环 以 及 包含 多 条 语句 的 让 代码 块 都 需要 使 用 大 括号 。 
如 果 代 码 行 数 较 多 ， 很 可 能 造成 大 括号 遗漏 。 

例 7.3 ”使 用 双 层 for 循环 输出 两 个 数 的 乘积 ， 并 得 到 内 层 循环 缺少 结束 大 括号 错误 。〈 实 例 位 置 : 光 
盘 \TMNInstance\07\7.3) 

<?php 

for($i=0; $i<10; $i++) 


{ 
142 


for($j=0; $j<10; $j++) 
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echo $i*$j."<br>"; 
// 遗 漏 循环 体 结束 大 括号 
} 


Dy: 

运行 上 述 实 例 ， 将 在 页 面 中 输出 如 图 74 所 示 的 错误 提 
示 。 从 错误 提示 中 可 以 得 知 内 层 嵌 套 无 结束 标志 ， 这 就 是 未 
写 内 层 for 循环 的 结束 大 括号 所 导致 的 。 

4. 缺少 变量 标识 符 $ 引 起 的 语法 错误 . . 

在 PHP 中 ， 设 置 变量 时 需要 使 用 美元 符号 “$”， 如 果 A 
不 添加 美元 符号 就 会 引起 解析 错误 。 1 

.4 缺少 for 循环 结束 大 括号 错误 提示 

例 7.4 “下 面 通过 实例 讲解 在 使 用 变量 时 不 添加 美元 符 图 74 似 少 如 特 bet 
号 会 产生 什么 错误 。〔 实 例 位 置 ， 光盘 \TM\Instance\07\7.4) 

<?php 


for ( $i=1; $i<11; i++ /缺少 一 个 变量 的 美元 符号 
echo $i."<br>"; 
b 


3 

运行 程序 时 ， 输 出 如 下 错误 提示 信息 : 

Parse error syntax error, unexpected T_INC, expecting ') in E:\AppServ\Wwww\07\7.4\index.php on line 2 

在 该 语句 中 ，“it+” 应 该 修改 为 “$iH”， 如 果 是 前 面 的 “$i<11” 没 有 使 用 美元 符号 ， 那 么 该 程序 将 
进入 无 限 循环 的 状态 直到 服务 器 终止 执行 。 

语法 的 错误 是 最 基本 的 错误 ， 只 要 在 编写 程序 时 认真 一 些 ， 就 会 减少 很 多 麻烦 。 这 里 介绍 的 只 是 语法 
错误 中 常见 的 几 种 类 型 ， 还 有 很 多 类 似 的 错误 ， 要 避免 错误 的 出 现 就 要 靠 广大 程序 设计 人 员 平 时 编写 代码 
时 多 加 注意 ， 尽 量 书写 准确 的 代码 。 


7.2.2 ”语义 错误 


语义 错误 是 在 语法 正确 的 前 提 下 导致 的 错误 。 例 如 ,应 用 PHP 连接 符 实现 两 个 字符 串 的 连接 。 代 码 如 下 : 
<?php 

$str="PHP 开发 实战 宝典 "; 

$url="www.mingribook.com"; 

echo $str + $url; // 错 误 地 使 用 了 字符 串 连 接 符号 ， 正 确 的 连接 符 是 “.” 
?> 
由 于 PHP 中 的 字符 串 连 接 符 是 “.”， 而 不 是 “+”。 上 面 的 代码 错误 地 使 用 了 “+” 作 为 字符 串 的 连接 

符 。 但 是 由 于 PHP 能 够 隐 式 转换 变量 类 型 ， 上 面 的 代码 并 不 会 导致 编译 器 出 错 ， 只 是 不 会 输出 正确 的 结果 。 


7.2.3 ”逻辑 错误 


逻辑 错误 对 于 PHP 编译 器 来 说 并 不 算 错误 ， 但 是 由 于 代码 中 存在 的 逻辑 问题 ， 导 致 运行 结果 没有 得 到 
期 望 的 结果 。 轨 辑 错误 在 语法 上 是 不 存在 错误 的 ， 但 从 程序 的 功能 上 看 是 缺陷 ， 它 是 最 难 调试 和 发 现 的 ， 
因为 它们 不 会 抛 出 任何 错误 信息 ， 唯 一 能 看 到 的 就 是 程序 的 功能 (或 部 分 功能 ) 没有 实现 。 

例如 ， 某 商城 实现 商品 优惠 活动 ， 如 果 用 户 为 普通 用 户 ， 那 么 商品 不 打折 ; 如 果 是 商城 的 会 员 ， 那 么 


商品 打 八 五 折 。 代 码 如 下 : 
@ 
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<?php 
if($user=" 普 通用 户 "X{ 
echo $price=485*1; /485 是 商品 价格 ，1 是 指 不 打折 
小 
if($user=" 会 员 "}{ 
echo $price=485*8.5; /485 是 商品 价格 ，8.5 是 指 打 八 五 折 
上 
?> 


上 面 的 代码 对 于 PHP 编译 器 来 说 没有 任何 问题 。 运 行程 序 时 ， 程 序 没有 弹出 错误 信息 。 但 是 当 用 户 为 
商城 的 会 员 时 ， 商 品 价格 乘 以 一 个 8.5， 这 一 点 就 没有 符合 要 求 ， 属 于 逻辑 错误 。 应 该 乘 以 0.85 才 正确 。 


jj/ 
ea 


该 再 除 以 10， 这 样 ， 就 相当 于 原来 的 商品 价格 乘 以 0.85。 正 确 的 代码 为 : 
echo $price=485*8.5/10; /1485 是 商品 价格 ，8.5 是 指 打 八 五 折 


对 于 多 辑 错误 而 言 ， 发 现 错误 是 容易 的 ， 但 要 查找 出 逻辑 错误 的 原因 却 很 困难 。 因 此 ， 在 编写 程序 的 
过 程 中 ， 一 定 要 注意 使 用 语句 或 者 函数 的 书写 完整 性 ， 否 则 将 导致 程序 出 错 。 


7.2.4 注释 错误 


培养 编写 注释 的 习惯 ， 对 于 一 个 程序 员 来 说 是 很 有 帮助 的 。 它 可 以 增强 程序 的 可 读 性 ， 便 于 对 程序 的 
修改 和 后 期 维护 。 虽 然 错误 的 注释 并 不 影响 程序 的 运行 ， 但 是 会 给 维护 人 员 在 后 期 的 维护 上 带 来 一 定 的 难 
度 。 例 如 : 

<?php 

$backTime=date("Y-m-d",(time()+3600*24*30)); 。 // 格 式 化 SbackTime 变量 为 系统 当前 日 期 


?> 


上 面 获 取 时 间 的 代码 与 后 面 的 注释 不 符 ， 注 释 应 改 为 “$backTime 为 当前 日 期 130 天 期 限 ”。 
7.2.5 ”运行 错误 


运行 错误 的 原因 不 容易 确定 ， 它 可 能 是 由 脚本 导致 的 ， 也 可 能 是 在 脚本 的 交互 过 程 中 或 其 他 的 事件 、 
条 件 下 产生 的 ， 这 就 需要 程序 员 平 时 多 积累 遇 错 处 理 的 方法 ， 以 提高 解决 问题 和 分 析 问 题 的 能 力 。 

下 面 的 几 种 情况 是 常见 的 运行 错误 : 

回 ”调用 不 存在 的 文件 。 在 编写 程序 时 ， 由 于 调用 文件 的 名 称 书写 错误 ， 导 致 调用 了 一 个 不 存在 的 
文件 。 

调用 不 存在 的 函数 。 在 编写 程序 时 ， 如 果 函 数 名 称 书写 错误 ， 此 时 就 会 产生 错误 。 即 使 函数 名 写 
对 了 ， 但 使 用 的 参数 不 对 ， 同 样 也 会 产生 一 个 错误 。 

读 写 文件 。 访 问 文件 的 错误 也 是 经 常 出 现 的 ， 如 硬盘 驱动 器 出 错 或 写 满 ， 人 为 操作 错误 导致 目录 
权限 改变 等 。 如 果 没 有 考虑 到 文件 的 权限 问题 ， 文 件 权限 设置 为 只 读 属 性 ， 直 接 对 文件 进行 操作 
就 会 产生 错误 。 由 于 该 文件 具有 只 读 的 权限 ， 不 能 进行 写 入 的 操作 。 在 执行 这 项 操作 时 ， 首 先 要 
明确 该 文件 的 属性 是 否 为 可 写 。 如 果 要 坚持 执行 操作 ， 那 就 需要 修改 文件 的 权限 。 

运算 的 错误 。 在 进行 一 些 算术 运算 或 者 逻辑 运算 的 过 程 中 ， 如 果 出 现 不 符合 运算 法 则 的 运算 ， 例 
如 ， 在 做 除法 运算 时 ， 分 母 为 0， 就 会 产生 错误 。 
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7.3 PHP 错误 的 调试 


PHP 最 基本 的 调试 策略 是 使 用 PHP 的 错误 报告 机 制 。 之 所 以 说 这 是 最 基本 的 调试 策略 ， 是 因为 对 PHP 
文件 的 设置 是 在 安装 时 就 必须 配置 好 的 ， 否 则 就 无 法 使 用 。 


7.3.1 PHP 的 错误 报告 


如 4 视频 讲解 : 光盘 \TM\Video\ 第 7 章 \PHP 的 错误 报告 .exe 
通常 在 一 个 PHP 脚本 中 发 生 错 误 时 ， 出 错 信息 会 被 插入 到 当前 脚本 进行 输出 ， 如 果 是 致命 的 错误 ， 脚 
本 就 会 终止 执行 。 

PHP 将 脚本 错误 分 成 提示 (notice) 、 警 告 (warning) 和 错误 (error) 3 个 等 级 。 

notice 信息 可 能 是 脚本 运行 错误 导致 的 ,也 可 能 只 是 在 正常 运行 过 程 中 出 现 的 。 实 际 上 这 也 许 就 是 
代码 中 的 缺陷 ， 因 为 PHP 对 其 解释 可 能 与 代码 的 本 义 有 所 不 同 。 

warming 标识 着 一 个 非 致命 性 的 错误 ， 在 代码 运行 时 产生 。 它 们 并 不 是 致命 的 错误 ， 不 会 停止 脚本 
的 执行 。 

error 消息 说 明 出 现 了 致命 的 错误 ， 会 导致 脚本 停止 运行 。PHP 运行 的 任何 阶段 都 可 能 产生 这 种 错 
误 ， 包 括 初始 化 、 解 析 和 执行 代码 阶段 。 


7.3.2 ”启动 错误 报告 


茹 4 视频 讲解 : 光盘 \TM\Video\ 第 7 章 \ 启 动 错误 报告 .exe 

使 用 PHP 的 错误 报告 是 调试 程序 的 基础 。 通 过 对 PHP 配置 文件 的 设置 ， 使 程序 在 执行 过 程 中 自动 抛 出 
鞭 误 的 代码 行 ， 帮 助 开发 者 发 现 并 消除 错误 。 当 项 目 发 布 后 ， 需 要 关闭 错误 报告 ， 用 户 不 能 看 到 错误 信息 ， 
这 样 非常 不 人 性 化 。 

启动 错误 报告 ， 也 就 是 修改 php.ini 文件 中 的 报错 设置 ， 具 体 设置 如 图 7.5 所 示 。 

display_errors 变量 的 目的 很 明显 ， 用 于 告诉 PHP 是 否 显示 错误 。 默认 值 是 Off。 但 是 ， 要 让 开发 过 程 更 
加 轻松 ， 需 要 把 这 个 值 设 为 On， 以 便 根据 错误 提示 调试 程序 。 

error reporting 变量 的 默认 值 是 E_ALL。 这 个 设置 会 显示 从 不 良 编码 实践 到 无 害 提示 的 所 有 信息 .E_ALL 
值 的 设置 对 于 开发 过 程 来 讲 过 于 细 化 。 通 常 只 需要 设置 错误 提示 和 显示 不 良 编码 即 可 ， 因 此 ， 需 要 把 这 个 
值 设置 为 “E_ALL &~E_NOTICE”。 

设置 完成 后 ， 保 存 php.ini 文件 ， 然 后 重新 启动 Apache 服务 器 即 可 。 

通过 这 些 错误 设置 ， 开 发 者 在 执行 程序 的 过 程 中 ， 就 可 以 在 浏览 器 中 查看 到 错误 信息 ， 以 及 出 错 代码 
的 行 号 ， 并 分 析 产 生 错误 的 原因 。 


7.3.3 ”使 用 print 语句 调试 程序 
区 4 视频 讲解 : 光盘 \TM\Video\ 第 7 章 \ 使 用 print 语句 调试 程序 .exe 


print 语句 用 于 字符 串 的 输出 。 基 于 print 语句 灵活 的 特点 ， 在 程序 调试 过 程 中 经 常 被 广泛 使 用 。 
例 7.5 ”通过 实例 介绍 如 何 应 用 print 语句 对 程序 进行 调试 。 在 本 实例 中 实现 一 个 获取 表单 提交 的 数据 


_ 国 
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功能 。〈 实 例 位置 ; 光盘 \TMNInstance\07\7.5) 
程序 代码 如 下 : 
<form name="form1" method="post" action="™"> 
<input name="txt_key" type="text" id="txt_key"> 
<input type="submit" name="Submit" value=" 搜 索 "> 
</form> 
<?php 
if($_POST[Submit]==" 搜 索 "X{ 
$key=$_POST[txt_key]; // 获 取 文本 框 的 值 
2 
上 面 的 代码 实现 了 获取 文本 框 的 值 ， 由 于 语句 中 没有 输出 ， 因 此 当 提 交 表 单 时 也 看 不 到 是 否 获取 了 信 
息 ， 这 时 可 以 应 用 print 语句 输出 文本 框 的 值 ， 以 此 来 判断 是 否 正确 获取 文本 框 的 值 。 在 加 粗 的 代码 下 添加 
如 下 语句 。 
print $key; // 输 出 文本 框 的 值 
print 语句 输出 了 POST 数组 中 的 数据 ， 即 文本 框 的 值 ， 如 果 数 组 中 的 数据 未 能 正确 显示 ， 则 说 明 没有 
正确 获取 到 文本 框 的 值 ， 从 而 断定 代码 中 有 错误 ， 需 要 查找 错误 源头 ， 并 加 以 修正 。 如 果 print 语句 输出 的 
字符 串 与 输入 到 文本 框 中 的 字符 串 相符 ， 则 说 明 程序 正确 。 
运行 结果 如 图 7.6 所 示 。 


diaplay_errora = 0 


ng = EALLR "ENOTICE 


ET 


7.5 设置 php.ini 文件 中 的 错误 处 理 机 制 图 7.6 使 用 print 语句 判断 是 否 正确 获取 到 文本 框 的 值 


使 用 print 语句 可 以 缩小 缺陷 可 能 出 现 的 范围 ， 能 够 最 终 确定 错误 的 位 置 或 验证 数据 信息 。 在 程序 中 熟 
练 地 使 用 print 语句 ， 可 以 迅速 找 出 程序 中 的 错误 根源 。 


7.3.4 应 用 @ 前 缀 字符 屏蔽 PHP 脚本 错误 提示 


多 视频 讲解 : 光盘 \TM\Video\ 第 7 章 \ 应 用 @ 前 缓 字符 屏蔽 PHP 脚本 错误 提示 .exe 

前 级 字符 @ 不 是 一 个 命令 ,而 是 DOS 批 处 理 的 一 个 特殊 标记 符 ， 是 命令 行 回 显 屏蔽 符 。 

前 缀 字符 @ 表 示 执 行 时 本 行 在 cmd 中 不 显示 , 可 以 使 用 echo off 命令 关闭 显示 , 仅 用 于 屏蔽 命令 行 回 显 。 
在 默认 情况 下 ， 当 PHP 代码 遇 到 错误 时 会 显示 错误 信息 ， 包 括 当前 脚本 的 绝对 路 径 ， 这 样 会 造成 一 些 不 安 
全 的 因素 ， 可 以 在 调用 函数 时 加 上 前 组 字符 @， 以 屏蔽 错误 信息 。 

例 7.6 当 试 图 打开 一 个 不 存在 的 文件 时 ， 屏 项 错误 信息 以 避免 路 径 泄露 。 (实例 位 置 ; 光盘 \TM 
Instance\07\7.6) 

代码 如 下 : 

<?php 

S$file="book.php"; /定义 操作 文件 

@fread($file) or die(" 文 件 读 取 失 败 ! "); 

fclose($file); 

echo "我 不 能 被 输出 了 ! 程序 运行 后 ， 前 缀 字符 @ 后 面 的 命令 不 被 显示 !"; 

a 


[Ca 
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fread0 函 数 前 应 用 了 前 组 字符 @， 说 明 程序 运行 时 在 浏览 器 上 不 会 显示 前 绥 字 符 @ 后 面 的 命令 。 
结果 为 : 文件 读 取 失 败 ! 


7.3.5 ”使 用 错误 处 理 器 记录 日 志 


PHP 提供 了 内 建 的 错误 处 理 函数 error log0， 可 以 将 出 错 信息 记录 到 管理 员 所 指定 的 路 径 。 
error_log(message,type [, destination, [,extra_headers]]); 
参数 message 指出 错 信息 ; 参数 type 指定 出 错 信息 记录 的 指 < - 
定位 置 。 如 果 用 PHP 的 日 志 记 录 机 制 保存 出 错 信 息 ， 需 要 将 参数 3 0) BR. A 
type 的 值 设置 为 0。 如果 将 错误 追加 到 destination 文件 中 ， 需 要 的 
将 参 9 值 设 置 为 3。 ， 需 要 修 .ini 
ws 人 ee 人 需要 修改 pbp ini 中 的 exor_Jog 图 77 设置 php ini 文件 中 的 错误 处 理 机 制 
设置 完成 后 ， 保 存 php.ini 文件 ， 然 后 重新 启动 Apache 服务 器 即 可 。 
例 7.7 实现 一 个 错误 处 理 的 例子 ， 将 日 志 记录 到 日 志文 件 中 ， 并 在 文件 大 于 1024B (1KB) 时 对 日 志 
文件 进行 重 命名 。〔 实 例 位 置 ， 光盘 \TM\Instance\07\7.7) 


代码 如 下 : 
<?php 
function err_log($error,$error_str}{ // 自 定义 一 个 错误 处 理 函 数 
$file="php_error.log"; 
if(filesize($file)>1024X{ // 如 果 日 志文 件 大 于 1024B 
rename($file,$file.(string)time()); /以 时 间 为 依据 对 日 志文 件 进行 重 命名 
clearstatcache(); // 清 除 文件 状态 缓存 
error_log($error_str'0,$file); // 将 出 错 信息 记录 到 管理 员 所 指定 的 路 径 
set_error_handler('err_log'); /| 执行 自 定义 函数 log_roller() 
trigger_error(time().": 程 序 报错 .\n"); /发 出 错误 信息 
restore_error_handler(); /重新 编译 这 个 预 错 处 理 的 函数 
?> 


在 上 面 的 代码 中 , 加 粗 的 PHP 内 置 错误 处 理 函数 error_log0 的 type 参数 值 设置 为 0, 将 出 错 信息 记录 在 
指定 的 文件 园 waen oo 中 ， 每 运行 一 次 程序 或 剧 新 一 次 ， 该 文件 将 即时 更 新 ， 将 每 次 的 出 错 信息 加 载 到 该 文 
件 中 。 


[3 技巧 如 果 将 参数 type 的 值 设置 为 3， 则 将 错误 追加 到 destination 文件 中 , 每 运行 一 次 或 刷新 一 次 程 
序 ， 都 会 生成 一 个 新 的 错误 日 志 。 


通常 ， 内 存 的 地 址 是 以 字 节 〈byte) 为 单位 编制 的 ， 内 存 的 容量 一 般 以 KB 或 MB 为 单位 。 

一 般 情 况 下 ， 在 开发 一 个 网 站 时 ， 为 了 调试 方便 ， 和 希望 错误 直接 显示 在 页 面 上 。 但 当 一 个 网 站 正式 发 
布 到 互联 网 上 时 ， 不 能 把 内 部 的 错误 信息 显示 给 访问 者 ， 最 佳 的 方法 是 记录 到 错误 日 志 中 。 这 时 ， 需 要 在 
php.ini 文件 中 做 如 下 设置 。 

display_errors = Off 

log_errors=on 

error_log=/tmp/errors.log 


设置 完成 后 ， 保 存 php.ini 文件 ， 然 后 重新 启动 Apache 服务 器 即 可 。 
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7.4 SQL 错误 的 调试 


SQL 语句 错误 也 是 相当 常见 的 ， 它 们 是 由 MySQL 数据 库 报告 的 。SQL 错误 主要 体现 在 两 个 方面 : 第 
一 是 它 和 PHP 的 连接 方面 ， 第 二 是 对 其 自身 的 使 用 上 ， 也 就 是 在 执行 MySQL 语句 时 的 错误 。 


7.4.1 PHP 与 MySQL 连接 错误 


区 中 视频 讲解 : 光盘 \TM\Video\ 第 7 章 \PHP 与 MySQL 连接 错误 .exe 
PHP 与 MySQL 数据 库 的 连接 是 进行 程序 开发 的 第 一 步 ， 如 果 不 能 成 功 地 把 它们 连接 起 来 , 工作 就 无 法 
进行 。 这 就 使 它们 连接 中 的 错误 成 为 首先 要 解决 的 问题 ， 也 是 必须 要 解决 的 问题 。 
在 通过 PHP 函数 连接 MySQL 数据 库 时 ， 有 可 能 出 现 的 错误 包括 : 
使 用 的 PHP 函数 不 正确 ， 不 过 这 种 情况 不 是 很 常见 ， 因 为 如 果 是 函数 书写 错误 ， 那 么 在 具备 高 亮 
代码 显示 功能 的 开发 工具 中 能 够 直接 看 出 函数 书写 是 否 正确 。 
使 用 函数 没有 问题 ， 但 输入 的 参数 不 正确 。 输 入 了 错误 的 数据 库 用 户 名 、 密 码 或 者 指定 的 数据 库 
不 存在 ， 同 样 会 导致 数据 库 连 接 失 败 。 
例 7.8 通过 PHP 函数 连接 db_database07 数据 库 ， 数 据 库 用 户 名 为 root， 密 码 为 root。〔 实 例 位 置 : 
光盘 \TMNVInstance\07\7.8) 
代码 如 下 : 
<?php 
$con=mysql_connect('localhost','root','root') or die(" 与 服务 器 连接 失败 "); 
echo "与 服务 器 连接 成 功 !<br>"; 
mysql_select_db('db_database07',$con) or die(" 没 有 找到 数据 库 ""); 
echo "成 功 "; 
?> 
本 实例 在 编写 代码 的 过 程 中 ,使 用 的 密码 不 是 root， 而 是 
111， 所 以 导致 数据 库 连 接 失败 ， 其 运行 结果 如 图 7.8 所 示 。 


ed 


Waming: ageql cocci ‘sql sanaeet} Aceeet denied for veer 3 
ro ehos Cosing pesswcrd YES) in Pr AgpSerowww TMOT,? Biloles ph cn te | 


语句 ， 它 是 PHP 的 一 种 错误 处 理 机 制 。 当 它 左 侧 的 语句 发 。 | Snrssexm” 


生 错 误 时 ，die0 语 句 会 显示 其 字符 串 参 数 的 内 容 ， 而 且 程 二 
序 会 退出 。 图 7.8 密码 不 正确 导致 连接 失败 


7.4.2 ”SQL 语句 错误 


区 4 视频 讲解 : 光盘 \TMIVVideo\ 第 7 章 \SQL 语句 错误 .exe 
编写 的 PHP 语句 和 数据 库 的 连接 都 没有 问题 ,错误 出 现在 SQL 语句 的 执行 过 程 中 ， 出 现 这 种 错误 的 原 
因 有 以 下 3 个 。 
数据 提交 页 面 与 数据 处 理 页 面 之 间 ，form 表单 中 定义 的 字段 的 名 称 与 数据 处 理 页 中 获取 字段 值 所 
使 用 的 名 称 不 一 致 ， 导 致 获取 不 到 数据 ， 如 图 7.9 所 示 。 
数据 处 理 页 中 执行 的 insert 语句 ， 参 数 的 数量 与 参数 值 的 数量 不 匹配 ， 或 者 参数 与 参数 值 的 顺序 不 


@® 
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匹配 ， 都 将 导致 添加 失败 ， 如 图 7.10 所 示 。 


EEC 


El ol 


密码 文本 域 的 字段 | 
名 称 : pass 和 、、 i 量 


i 


LE 


图 7.9 数据 提交 页 与 数据 处 理 页 使 用 的 字段 不 统一 图 7.10 数据 处 理 页 中 出 现 的 错误 


加 


数据 处 理 页 insert 语句 的 参数 与 数据 表 中 字段 不 匹配 ,使 得 字段 名 称 不 一 致 、 字 段 数量 不 匹配 或 者 
数据 类 型 不 一 致 ， 都 将 导致 添加 失败 ， 如 图 7.11 所 示 。 


解决 SQL 语句 中 错误 的 方法 很 多 ， 这 里 介绍 3 种 比较 常用 的 方法 。 


[al 


回 


将 SQL 语句 复制 到 对 应 的 数据 库 中 , 在 数据 库 图 形 化 管理 工具 中 的 SQL 语言 下 执行 。 如 果 可 以 执 
行 ， 那 么 说 明 该 语句 是 正确 的 ， 如 不 能 执行 ， 则 说 明 是 语句 本 身 存在 错误 ， 就 可 以 根据 数据 库 图 
形 化 管理 工具 给 出 的 错误 信息 ， 分 析出 错误 原因 。 

使 用 mysql_error0 语 句 输出 错误 信息 , 通过 该 语句 可 以 像 语 法 错误 那样 返回 一 个 错误 信息 。 该 语句 
的 使 用 被 放置 于 mysql_query0 语 句 的 后 面 。 如 果 将 die0 语 句 与 mysql_error0 语 句 组 合 应 用 ， 当 程 
序 结束 时 就 会 显示 MySQL 的 错误 信息 。 

将 SQL 语句 定义 成 一 个 单独 的 变量 ， 应 用 echo0 语 句 输出 这 个 变量 ， 分 析 这 个 变量 输出 的 值 是 否 
正确 ， 如 果 正 确 则 说 明 SQL 语句 没有 问题 。 


例 7.9 通过 PHP 函数 连接 db_database07 数据 库 ， 数 据 库 用 户 名 为 root， 密 码 为 111， 向 user 表 中 添 


-条 记录 ， 并 通过 die0 语 句 与 mysql_error0 语 句 返回 错误 信息 。〔 实 例 位 置 ， 光盘 \TMNInstance\07\7.9) 


代码 如 下 : 
<!-- 处 理 文件 代码 --> 
<?php 


了 


$conn=mysql_connect(localhost,root,"111") or die(" 与 服务 器 连接 失败 史 ); 
mysql_select_db('db_database07',$conn) or die(" 没 有 找到 数据 库 ""); 
if($Submit==" 提 交 "X{ 


$admin=$_POST[admin]; /获取 提交 的 用 户 名 
$pass=$_POST[pass]; /获取 提交 的 密码 
$dates=date("Y-m-d Hii:s"); /定义 当前 时 间 


$query="insert into user (admin,password,dates) values('$admin','$pass','$dates')"，// 编 写 添加 语句 
$result=mysql_query($query,$conn) or die(mysql_error()); 。 ”// 执 行 添 加 语句 ， 并 返回 错误 信息 
if($result}{ 
echo "<script>alert(' 添 加 成 功 ! '); window.location.href='index.php';</script>"; 
}else{ 
echo "添加 失败 路; 
} 


} 


<!- 表 单 提交 页 ， 设 置 的 表单 元 素 -> 


全 
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<form name="form1" method="post" action="index.php"> 
<input name="admin" type="text" id="admin" size="20"> 
<input name="pass" type="password" id="pass" size="20"> 


<input type="submit" name="Submit" value=" 提 交 "> 
</form> 


本 实例 中 错误 的 原因 是 insert 语句 的 参数 与 数据 表 中 的 字段 名 称 不 匹配 , 在 数据 表 中 没有 找到 指定 的 字 


段 ， 其 运行 结果 如 图 7.12 所 示 。 


) or ds(" 与 及 务 各 这 桂 夫 败 1-) 
(" 视 有 执 到 数据 库 1") ; 


一 


| pass_ |] warenars 四 ob231?_ehirese_el 
tates dume 时 间 


NE 和 
De ert 『 深 吉成 功 ! [nsent 语句 的 参数 名 称 ， Passwoad-coo， 
echo “ 注 与 数据 表 中 字段 名 称 :pass 不 匹配 ， 
人 导致 添加 失败 


Bmp/fecalhor se 


图 7.11 insert 语句 的 参数 与 数据 表 中 字段 名 称 不 匹配 


图 7.12 SQL 语句 中 出 现 的 错误 


a 
A 培 明 上 面 介绍 的 只 是 SQL 语句 中 的 一 种 错误 情况 ， 至 于 在 select、update 和 delete 等 语句 的 执行 过 


程 中 的 错误 需要 读者 自己 去 研究 与 分 析 。 


7.5 实 
敬 i 视频 讲解 : 光盘 \TMVVideo\ 第 7 章 \ 实 战 .exe 


7.5.1 运行 缺少 第 三 方 组 件 的 程序 


战 


在 运行 光盘 中 涉及 到 第 三 方 组 件 的 程序 时 ， 由 于 没有 看 光盘 中 提供 的 使 用 说 明 书 ， 而 直接 运行 程序 ， 


那么 将 导致 程序 运行 出 错 。 


例 7.10 在 这 个 实例 中 ， 通 过 Jpgraph 类 库 生 成 折线 图 ， 分 析 2010 年 公司 销售 额 ， 并 且 在 折线 图 中 通 
过 插入 的 图 像 设置 数据 点 标签 。 但 是 在 直接 运行 光盘 中 的 程序 时 ， 却 出 现 了 如 图 7.13 所 示 的 错误 ， 其 原因 
就 是 没有 认真 的 阅读 程序 使 用 说 明 书 ， 因 为 本 程序 应 用 到 一 个 第 三 方 的 组 件 Jpgraph 类 库 ， 需 要 自行 下 载 ， 并 
且 将 下 载 的 sre 文件 夹 复制 到 实例 的 根 目录 下 , 然后 才 可 以 运行 程序 。( 实 例 位 置 : 光盘 \TMNInstance\07\7.10) 


7.5.2 ”通过 readfile() 函 数 访问 远程 文件 


例 7.11 在 本 实例 中 通过 文件 系统 函数 库 中 的 readfile0 函 数 访问 一 个 远程 文件 ， 但 是 由 于 本 机 php.ini 


人 后 
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文件 的 配置 ， 不 允许 访问 远程 文件 ， 结 果 导 致 程序 运 行 失败 。《〈 实 例 位 置 ， 光盘 \TMNImstance\07\7.11) 


代码 如 下 : 
<?php 
if(@readfile('http://192.168.1.235/ ,7)X // 判 断 文件 打开 的 操作 是 否 执 行 成 功 
Jelse{ 
echo "<script>alert(' 您 没有 访问 远程 文件 的 权限 ! ');window.location.href='error.html'</script>"; 
// 重 定向 页 面 
加 


如 果 想 让 服务 器 支持 远程 文件 的 访问 ， 必 须 修 改 php.ini 文件 ， 定 位 到 allow_url fopen = OFF 选项 ， 将 
allow_url fopen 的 值 修改 为 ON， 保存 后 重新 启动 Apache 服务 器 即 可 。 
运行 结果 如 图 7.14 所 示 。 


hep ocalhore a TMT 10) 


i si i 仆 、 术 服 务 吕 不 支持 远程 文件 的 访问 功能 


per 
sere TM 人 一 一 


(15995-2010 二 新 省 届 日 科技 有 有限 公司 


图 7.13 没有 找到 指定 的 包含 文件 图 7.14 服务 器 不 支持 远程 文件 的 访问 
7.5.3 ”解决 数据 库 乱码 问题 


在 通过 PHP 对 MySQL 数据 库 中 的 数据 进行 操作 的 过 程 中 , 有 时 输出 MySQL 数据 库 中 的 数据 会 出 现 乱 
码 。 这 是 一 个 让 人 十 分 头疼 的 问题 ， 纠 其 根源 该 问题 出 现 的 原因 是 数据 库 中 数据 对 字符 编码 格式 的 限制 。 
由 于 在 创建 数据 库 时 使 用 的 编码 格式 是 “utf8”， 所 以 在 对 数据 库 中 的 数据 进行 输出 或 者 添加 过 程 中 必须 使 
用 相同 的 编码 格式 才 行 ， 否 则 就 出 现 乱 码 。 

例 7.12 在 本 实例 中 , 通过 while 语句 循环 输出 数据 库 中 数据 ， 比 较 在 设置 数据 库 编码 格式 与 不 设置 数 
据 库 编码 格式 时 输出 数据 的 不 同 之 处 。《〈 实 例 位置 ; 光盘 \TMNInstance\07\7.12) 

(1) 创建 数据 库 连 接 文件 conn.php， 完成 与 MySQL 服务 器 的 连接 , 用 户 名 是 root， 密 码 为 111。 然后 ， 
连接 db_database07 数据 库 。 最 后 设置 数据 库 编码 格式 为 utfg 。 其 关键 代码 如 下 : 


<?php 

$conn=mysql_connect("localhost","root","111") or die(" 服 务 器 连接 失败 :".mysql_error()); /| 连接 服务 器 
mysql_select_db("db_database07",$conn) or die ("数据 库 连 接 失 败 :".mysql_error()); // 连 接 数 据 库 
mysql_query("set names utf8"); // 设 置 数 据 库 编 码 格 式 

2 


(2) 创建 index.php 文件 。 首先 通过 include_onceO 语 句 包含 数据 库 连接 文件 ,然后 定义 SQL 查询 语句 ， 
最 后 执行 SQL 语句 ， 并 且 通 过 while 语句 完成 数据 库 中 数据 的 循环 输出 。 其 关键 代码 如 下 : 


<?php 

include_once("conn/conn.php"); // 包 含 数 据 库 连 接 文 件 
$sql="select * from tb_log "; /定义 SQL 语句 
$query=mysql_query($sql,$conn); /执行 查询 操作 
echo mysql_error(); // 返 回 错误 信息 ， 如 果 存 在 
While($myrow=mysql_fetch_array($query))f /循环 输 出 查询 结果 
3 

<tr> 


<td><?php echo $myrow[id]?></td> 
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<td><?php echo $myrowflog_name']?></td> 
<td><?php echo $myrow[log_date'];?></td> 


</tr> 
<?php 
} 
2> 
运行 效果 如 图 7.15 和 图 7.16 所 示 。 
加 标题 时 间 加 标题 时 间 
1 数据 库 服 务 器 问题 2012-11-120831: 和 9 1 Ee 2012-11-12083149 
2 效 据 表 问题 2012-11-12083223 2 E22 2012-11-12083223 
5 槛 方法 2012-11-1407:5257 5 EE 2012-11-14073237 
图 7.15 正常 的 数据 输出 结果 图 7.16 数据 输出 乱码 


7.5.4 封装 异常 处 理 类 


例 7.13 在 程序 调试 中 ， 可 以 应 用 try…catchf} 语 句 捕获 错误 信息 ， 并 且 通 过 Exception 内 置 类 来 返回 
相应 的 错误 信息 。 在 此 基础 上 还 可 以 使 用 继承 对 Exception 类 进行 扩展 ， 编 写 属于 自己 的 异常 处 理 类 。 其 方 
法 就 是 ， 编 写 一 个 子 类 来 继承 Exception 类 ， 这 样 在 子 类 中 就 继承 了 父 类 的 所 有 属性 和 方法 ， 并 且 还 可 以 添 
加 子 类 所 特有 的 属性 和 方法 。 在 本 项 目 中 编写 一 个 可 以 判断 电话 号 码 格 式 是 否 正确 的 类 ， 当 定义 的 电话 号 
码 格式 不 正确 时 跳 转 到 自 定义 的 错误 页 面 , 并 且 输 出 错误 提示 信息 。( 实 例 位 置 : 光盘 \TM\Instance\07\7.13) 

(1) 封装 电话 号 码 格式 判断 的 异常 处 理 类 TelException， 继 承 Exception 类 。 其 关键 代码 如 下 : 

class TelException extends Exception{ /定义 TelException 类 ， 继 承 Exception 类 

public function errorTel(}{ /定义 方法 返回 错误 信息 
SerrorMsg = "出 错 原因 : ".$this->getMessage()." 不 是 一 个 合法 的 电话 号 码 "; 
SerrorMsg .="<br>"; 
$errorMsg .=" 错 误 文件 路 径 :".$this->getFile(); 
S$errorMsg .="<br>"; 
$errorMsg .=" 错 误 代码 行 号 :".$this-> getLine(); 
return $errorMsg; 


} 


(2) 创建 自 定义 函数 check_tel0， 通 过 正则 表达 式 和 preg_match0 函 数 验 证 电话 号 码 的 格式 是 否 正 确 。 
自 定义 函数 的 语法 如 下 : 


function check_tel($tel){ // 自 定义 函数 验证 电话 号 码 格式 是 否 正确 
S$checkphone="/^13(\d{9})$/"; // 定 义 验证 手机 号 码 的 正则 表达 式 
$counts=preg_match($checkphone, $tel); // 执 行 验证 操作 
return $counts; // 返 回 验证 结果 

} 

(3) 定义 被 验证 的 电话 号 码 ， 应 用 自 定义 异常 处 理 类 对 电话 号 码 格式 进行 验证 。 其 代码 如 下 : 

S$tel = "1371234****"; /定义 被 验证 的 电话 号 码 

六 
通过 自 定义 异常 处 理 类 返回 错误 提示 


a 


{ 
if(check_tel($tel) (=1X{ 
throw new TelException($tel); 


} 
}catch (TelException $e}{ 


include_once("error.php"); 
出 


@ 
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其 运行 结果 如 图 7.17 所 示 。 
明和 SEE aogl 


1 不 是 一 个 辣 生 外 各 尖 中 
AppserWrew 9 13umdecFhp 
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图 7.17 判断 电话 号 码 的 格式 是 否 正确 
7.5.5 ”解决 程序 的 语法 错误 
在 应 用 直 for、while 和 switch 等 流程 控制 语句 时 ， 必 须 注意 其 中 大 括号 的 书写 ， 它 们 必须 是 成 对 出 现 


的 ， 和 否则 将 导致 程序 运行 出 错 。 (实例 位 置 : 光盘 \TMNInstance\07\7.14) 
例 7.14 在 本 实例 中 ， 实 现 一 个 简单 的 网 页 计数 器 的 功能 ， 并 且 将 数据 存储 到 指定 的 文本 文件 中 。 其 


<?php 

if(($fp=fopen("counter.txt","r"))==false}{ 
echo "打开 文件 失败 上 

Jelse{ 
$counter=fgets($fp,1024); // 读 取 文 件 中 数据 
fclose($fp); /关闭 文本 文件 
$counter++; // 计 数 器 增加 1 
$fp=fopen("counter.txt","w"); /以 写 的 方式 打开 文本 文件 
fputs($fp,$counter); // 将 新 的 统计 数据 增加 1 
fclose($fp); 

if(($fp=fopen("counter.txt","r"))==falseX{ /打开 文件 
echo "打开 文件 失败 

Jelse{ 
Scounter=fgets($fp,1024); // 读 取 文 件 中 数据 
fclose($fp); /关闭 文件 

echo $counter; 

上 


运行 效果 如 图 7.18 所 示 。 


Paase em yntes eros, varspactd Seo a EA Semvanme TNEOTY.L4tndergly oalay30 


不 地 inranst | 人: 生 用 而 太 100% ~ 


图 7.18 由 于 大 括号 不 完整 导致 的 错误 
7.6 小 结 


本 章 主 要 介绍 了 程序 调试 及 异常 处 理 的 一 般 步骤 ， 并 着 重 讲解 了 PHP 中 常见 的 错误 及 其 原因 。 通 过 本 
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章 的 学 习 ， 初 学 者 可 以 掌握 PHP 中 常见 错误 的 处 理 方式 ， 并 能 够 积累 基本 的 程序 调试 经 验 与 技巧 。 希 望 读 
者 朋友 能 够 以 本 章 介 绍 的 内 容 为 基石 ， 进 一 步 拓展 与 积累 ， 最 终 登 上 编程 工程 师 的 舞台 。 


7.7 学 习 成 果 检 验 


1. 连接 不 存在 的 数据 库 ， 在 运行 光盘 中 涉及 到 数据 库 的 程序 时 ， 经 常会 犯 这 样 的 一 个 错误 ， 就 是 没有 
进行 数据 库 的 附加 ， 而 直接 运行 程序 ， 则 导致 输出 错误 信息 。 《实例 位 置 : 光盘 \TM\Instance\07\7.15) 

2. 在 开发 网 站 时 ， 为 了 调试 方便 ， 将 错误 直接 显示 在 页 面 上 。 但 当 一 个 网 站 正式 发 布 到 互联 网 上 时 ， 
就 不 能 把 内 部 的 错误 信息 显示 给 访问 者 ,其 最 佳 的 方法 应 该 是 记录 到 错误 日 志 中 。 实 现 一 个 错误 处 理 的 例子 ， 
将 日 志 记 录 到 日 志文 件 中 ， 并 在 文件 大 于 1024B (1KB) 时 对 日 志文 件 进行 重 命名 。〔 实 例 位 置 ， 光 盘 \TM\ 
Instance\07\7.16) 
ein 
error logO 函 数 的 语法 如 下 : 

error_log(message,type [, destination, [,extra_headers]]); 

参数 message 指出 错 信息 ; 参数 type 指定 出 错 信息 记录 的 指定 位 置 。 如 果 用 PHP 的 日 志 记录 机 制 保 
存 出 错 信息 ， 需 要 将 参数 type 的 值 设置 为 0; 如 果 将 错误 追加 到 指定 的 文件 中 ， 需 要 将 参数 type 的 值 设 
置 为 3。 

将 错误 日 志 存储 到 指定 文件 中 ， 不 在 页 面 中 显示 ， 还 需要 对 php.ini 文件 做 如 下 设置 : 

display_errors = Off 

设置 完成 后 ， 保 存 php.ini 文件 ， 然 后 重新 启动 Apache 服务 器 即 可 。 
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综合 实例 (一 ) 在 线 论 坛 
( 捉 视频 讲解 : 2S 分 钟 ) 


论坛 系统 已 经 成 为 时 下 比较 普遍 的 一 种 网 上 交流 手段 ， 现 在 很 多 
网 站 都 拥有 自己 的 论坛 ， 特 别 是 对 于 大 型 的 网 站 ， 论 坛 系统 已 经 成 为 
一 个 不 可 缺少 的 组 成 部 分 。 本 章 将 详细 介绍 论坛 系统 的 开发 、 基 本 操 
作 流 程 和 创作 思路 ， 为 读者 提供 一 个 整体 的 开发 思路 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

Wm 熟悉 用 户 注册 模块 的 设计 

H 熟悉 用 户 登 录 模 块 的 设计 

H 熟悉 帖子 分 类 管理 模块 的 设计 

让 了 解 发 帖 模块 设计 

让 了 解 回帖 模块 设计 

H 熟悉 后 台 管理 模块 设计 
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8.1 在 线 论坛 概述 


区 il 视频 讲解 :光盘 \TM\Video\ 第 8 章 \ 在 线 论坛 -exe 
本 章 开 发 的 是 一 个 最 基本 、 最 简单 的 论坛 系统 ， 其 具备 了 论坛 系统 的 基本 功能 ， 没 有 附加 任何 复杂 的 
功能 ， 完 全 适合 初学 者 学 习 和 研究 。 论 坛 系统 的 具体 功能 如 下 : 
MySQL 数据 库 的 创建 。 
用 户 注册 。 
用 户 登 录 。 
帖子 的 分 类 管理 。 
发 布 帖子 。 
回复 帖子 。 


后 台 管理 。 
8.1.1 程序 业务 流程 


在 线 论坛 系统 的 操作 流程 非常 清晰 ， 总 体 上 由 两 大 


加 回回 回回 罗 加 


模块 组 成 ， 前 台 展示 区 和 后 台 管理 ， 其 中 前 台 展示 区 的 es 
主要 功能 包括 用 户 注册 、 用 户 登 录 、 发 布 帖子 、 回 复 由 
子 等 ， 后 台 管 理 模块 的 主要 功能 包括 用 户 管理 、 栏 目 管 Wa 


理 、 主 题 管 理 、 回 复 内 容 管 理 和 非法 信息 管理 。 程 序 流 
程 如 图 8.1 所 示 。 


8.1.2 ”系统 预览 


在 线 论坛 由 多 个 页 面 组 成 ， 下 面 仅 列 出 几 个 典型 页 图 81 在 线 论坛 系统 操作 流程 图 
面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 

论坛 首页 面 的 运行 效果 如 图 8.2 所 示 。 

用 户 注册 页 面 的 运行 效果 如 图 8.3 所 示 ， 主 要 功能 是 实现 用 户 注册 。 


晤 岁 装 沪 
渍 只 严 藤 


消 叶 癌 访 弟 榜 


wonrsofr™~ sw” 


图 8.2 论坛 首页 运行 效果 8.3 用 户 注册 页 面 运行 效果 
发 布 主题 页 面 的 运行 效果 如 图 8.4 所 示 ， 主 要 功能 是 实现 发 布 帖子 。 
后 台 首 页 的 运行 效果 如 图 8.5 所 示 。 
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ER: FF 


EA pg 到 :二 


图 8.4 发 布 主题 页 面 运行 效果 


TE 


图 8.5 后 台 首 页 运行 效果 
8.2 ”数据 库 设计 
区 il 视频 讲解 : 光盘 \TM\Video\ 第 8 章 \ 数 据 库 设计 .exe 


8.2.1 数据 库 概要 说 明 


在 线 论坛 系统 中 ， 采 用 的 是 MySQL 数据 库 ， 用 来 存储 [ER 
用 户 信息 、 发 帖 信息 、 回 帖 信息 等 。 这 里 将 数据 库 命名 为 


下 fr ET 
db_forum， 其 中 包含 的 数据 表 如 图 8.6 所 示 。 oem 四 回头 国 X En 
加 F oy 
站 一 一] 三 加 加 于 山头 0 SA uncode ol 
人 S Teoresen 图 相 四 镁 转 X 1 MAEM UnB_unlc0ge_ol 
8.2.2 ”数据库 概念 设计 ee 
am 图 国 四 王国 X wsml ono 
5 个 家 时 23 MASAM_ wre_general cl 


根据 业务 流程 和 系统 功能 结构 ， 规 划 出 系统 中 使 用 的 数 图 8.6 数据 库 结构 
据 库 实体 对 象 及 实体 E-R 图 。 在 创建 数据 表 前 ， 首 先 需 要 创 
建 基本 信息 的 数据 表 ， 如 图 像 信息 表 、 版 主 信息 表 、 管 理 员 信息 表 等 。 

图 像 信息 表 实 体 E-R 图 如 图 8.7 所 示 。 

版 主 信息 表 实 体 E-R 图 如 图 8.8 所 示 。 


注册 日 期 
用 户 名 
管理 类 别 


8.8 版 主 信息 表 实 体 E-R 


图 8.7 图 像 信 息 表 实 体 ER 图 
管理 员 信息 表 实 体 E-R 图 如 图 8.9 所 示 。 


当 用 户 注 册 后 ， 需 要 将 用 户 信息 存储 到 数据 库 中 ， 包 括 用 户 的 用 户 名 、 真 实 姓名 、 密 码 、 住 址 等 属性 ， 
其 实体 E-R 图 如 图 8.10 所 示 。 
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图 8.9 管理 员 信息 表 实体 E-R 图 图 8.10 用 户 信息 表 实 体 E-R 图 

当 用 户 发 布 帖子 信息 时 ， 需 要 将 帖子 信息 存储 到 数据 库 中 ， 包 括 帖子 类 别 、 帖 子 主题 、 帖 子 内 容 等 ， 
其 实体 E-R 图 如 图 8.11 所 示 。 

当 用 户 回帖 时 ， 需 要 将 回帖 信息 存储 到 回帖 表 中 ， 包 括 回帖 主题 、 回 帖 内 容 、 回 帖 人 、 原 帖 主题 等 ， 
其 实体 E-R 图 如 图 8.12 所 示 。 


图 8.11 发 帖 实体 E-R 图 图 8.12 ”回帖 实体 E-R 图 
8.2.3 ”数据库 逻辑 设计 


在 线 论 坛 系统 是 典型 的 数据 库 开 发 应 用 程序 ， 论 坛 的 数据 库 设 计 是 一 个 非常 关键 的 环节 ， 下 面 将 对 本 
论坛 系统 中 使 用 的 数据 库 进行 介绍 。 

论坛 系统 中 创建 的 数据 库 名称 是 db_forum，MySQL 数据 库 服务 器 的 用 户 名 是 root， 密 码 是 81。 在 创建 
的 数据 库 中 包括 6 个 数据 表 ， 其 中 各 数据 表 实 现 的 功能 如 表 8.1 所 示 。 


表 8.1 db_forum 数据 库 中 数据 表 功能 说 明 


数据 表 名 称 功能 说 明 
tb_admin 管理 员 信息 表 ， 存 储 管理 员 的 个 人 信息 
tb_category 论坛 栏目 信息 表 ， 存 储 论坛 中 创建 的 栏目 信息 
tb_content 


发 布 帖子 信息 表 ， 存 储 用 户 在 论坛 中 发 布 的 帖子 信息 
表情 图 存储 表 ， 存 储 在 论坛 中 使 用 的 表情 图 

回复 帖子 信息 表 ， 存 储 对 论坛 中 帖子 的 回复 内 容 
注册 用 户 的 个 人 信息 表 ， 存 储 注册 用 户 的 个 人 信息 


tb_expression 


tb _ resume contents 


tb user 


和 ot 二 在 创建 数据 库 的 过 程 中 一 定 要 注意 字符 集 的 使 用 ， 要 选择 使 用 utf-8 类 型 ， 如 果 使 用 其 他 字符 
集 ， 有 可 能 会 导致 数据 库 中 的 数据 出 现 乱 码 。 


@ 
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这 里 使 用 了 6 个 数据 表 ， 其 中 各 个 表 的 结构 不 ene Dm 
再 一 一 介绍 。 下 面 以 用 户 信息 表 为 例 ， 来 了 解 一 下 tt 二 
数据 表 的 创建 过 程 ， 以 及 其 中 需要 注意 的 问题 。 a CEPROOEE 
tb_user 用 户 信息 表 的 结构 如 图 8.13 所 示 。 ee ee 人 此 明志 基本 更 季风 
在 使 用 MySQL 数据 库 创建 数据 表 时 ， 首 先 要 | 上 吕 wae 名 二 由 四 户头 加 加 加 
指定 一 个 字段 为 数据 表 主键 , 其 类 型 为 int (如 用 户 、 | ee 六 
表 中 的 ia) ， 然 后 在 创建 其 他 字段 时 ， 要 根据 字段 ”| woe 二 人 本 
表述 的 内 容 为 字段 定义 类 型 ， 例 如 ， 表 示 时 间 的 字 | wemeen 轩 本 人 的 时 


段 可 以 使 用 date 或 者 datetime 等 时 间 类 型 ， 而 表述 - 
大 量 的 文本 字段 时 ， 应 该 使 用 text 类 型 ， 如 果 存 储 国 819 bgs 攻关 作息 家 的 结构 

的 是 二 进 制 数据 ， 那 就 要 定义 blob 或 者 longblob 类 型 ， 具 体 字段 使 用 什么 样 的 类 型 来 定义 ， 要 根据 具体 问 
题 具 体 分 析 。 


入 站 在 创建 数据 表 时 ， 一 定 要 指定 数据 表 的 类 型 为 MyISAM， 如 果 使 用 其 他 类 型 将 影响 数据 库 的 
备份 ， 例 如 ， 使 用 InnoDB 类 型 保存 数据 ， 如 果 将 表 中 的 数据 复制 到 其 他 机 器 的 数据 库 中 ， 该 数据 表 将 
不 可 用 。 


8.3 ”用户 注册 模块 设计 
贸 g 视频 讲解 : 光盘 \TMVVideo\ 第 8 章 \ 用 户 注册 模块 设计 .exe 
8.3.1 用户 注册 模块 概述 
当 用 户 第 一 次 登录 本 论坛 时 ， 必 须 先 进行 注册 ， 然 后 登录 ， 才 可 以 发 表 帖 子 。 单 击 “ 注 册 ” 按 钮 ， 进 


入 到 注册 页 面 ， 按 照 要 求 输入 注册 信息 ， 单 击 “ 确 认 提交 ”按钮 ， 即 可 成 功 注 册 账 号 。 用 户 注册 模块 运行 
效果 如 图 8.14 所 示 。 


下 电网 上 存 绕 论坛 


用 户 注册 模块 的 运行 结果 
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8.3.2 ” JavaScript 脚本 和 include() 包 含 语句 


1. JavaScript 脚本 的 应 用 
在 论坛 的 用 户 注册 模块 中 ， 主 要 表述 的 是 用 户 注册 信息 ， 所 以 在 进行 页 面 设计 时 ， 重 点 把 握 如 何 设计 
表单 元 素 的 输出 方式 ， 以 及 如 何 对 表单 中 提交 的 数据 进行 处 理 。 在 对 表单 提交 的 数据 进行 处 理 的 过 程 中 ， 
使 用 JavaSeript 脚本 和 正则 表达 式 对 表单 中 提交 的 数据 进行 判断 , 以 及 对 用 户头 像 进 行 选择 。 主要 代码 如 下 : 
<script language="javascript">// 通 过 下 拉 列 表 选 择 头像 时 应 用 该 函数 
function showlogo(}{ 


document.images.img.src="images/tx/"+ 
document.form1 .tx.options[document.form1 .tx.selectedIindex].value; 


</script> 
2. include() 包 含 语句 的 应 用 
为 了 便于 对 论坛 中 头 尾 文件 进行 修改 , 这 里 充分 发 挥 了 include0 包 含 语句 的 作用 ， 头 尾 文件 都 是 通过 包 
含 语句 直接 调用 的 ， 无 须 在 本 页 编辑 。 
(1) 应 用 include 语句 包含 数据 库 服务 器 连接 文件 conn.php、 论 坛 的 头 文件 index_01.php 和 论坛 的 用 户 
登录 文件 index_02.php。 代 码 如 下 : 
<?php 
include("conn/conn.php"); 
include("index_01.php"); 
include("index_02.php"); 
让 
(2) 将 美工 设计 的 网 页 效果 图 嵌入 用 户 注册 信息 页 面 ， 应 用 表格 技术 合理 地 划分 和 设计 表单 的 布局 ， 
在 对 应 的 表格 中 嵌入 表单 元 素 ， 并 且 设 置 表单 元 素 的 名 称 。 


te 
设计 的 名 称 一 定 要 与 表单 处 理 页 中 使 用 的 表单 元 素 的 变量 名 称 相 吻 合 ， 否 则 将 导致 上 传 数 据 失败 。 


用 户 注册 模块 中 涉及 到 的 HTML 表单 元 素 如 表 8.2 所 示 。 
表 8.2 用户 注册 页 面 涉及 到 的 HTML 表单 元 素 
名 称 重要 属性 

form 1 method="post" action="login_ok.php" onSubmit="javascript: return checkitO:” 
Usermame, 注册 用 户 名 | <input name="username" type="text" id="username" /> 
true_name <input name="true name" type="text" id="true name" /> 

<input name="password" type="password" id="password" /> <span class="STYLE1"> 
zc_password 

*</span> 

<input name="password2" type="password" id="password2" /> <span class="STYLE1"> 
password2 

*</span> 

<input name="sex" type="radio" value=" 男 " checked="checked" /> 男 
<input type="radio" name="sex" value=" 女 "/> 女 
tel <input name="tel" type="text" id="tel" /> 
qd 
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续 表 


重要 属性 


<select size"l" id="tx" name="tx" onChange="showlogo0"> 


tx select 
email Text 


Text 


8.3.3 用 户 注册 模块 的 实现 过 程 


1. 页 面 设计 
本 实例 中 设计 的 用 户 注册 模块 包括 3 部 分 内 容 : 论坛 的 头 文件 、 论 坛 中 表单 元 素 的 详细 设置 和 论坛 中 
尾 文件 的 设置 。 用 户 注册 模块 的 设计 效果 如 图 8.15 所 示 。 


Mm | NE 


图 8.15 用 户 注册 模块 的 设计 效果 
2. 代码 设计 
在 完成 用 户 注册 模块 的 整体 布局 设计 后 ， 接 着 实现 用 户 注册 模块 的 功能 。 首 先 通过 JavaScript 脚本 对 表 
单 中 提交 的 数据 进行 判断 。 关 键 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\08\login.php) 
<script language="javascript"> 
function checkit(X{ 
if(form1.username.value=="™", 
alert(" 请 输入 用 户 名 !"); 
form1.username.select(); 
return(false); 


if(form1.password.value=="™")/{ 
alert(" 请 输入 用 户 密码 "); 
form1.password.select(); 
return(false); 


} 
se // 省 略 部 分 代码 
if(Iicheckemail(form1.email.value)X{ 
alert(" 邮 箱 地 址 格式 不 正确 M"); 
form1.email.select(); 
return(false); 
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return(true); 
} 
function checkemail(email){ // 验 证 邮箱 地 址 格式 是 否 正确 
var strs=email; 
var Expression=Nw+([-+.]Ww+)@Ww+([.JWw+)\ Wr([-.JWw+)*/; 
var objExp=new RegExp(Expression); 
if(objExp.test(strs)==trueX{ 
return true; 
}else{ 


return false; 
} 
} 


function checkphone(tel}{ // 验 证 电话 号 码 格式 是 否 正确 
var str=tel; 
var Expression=/^(\d{3}-)(\d{8})$I^(\d{4}-)(\d{7})$I^(d{4}-)(\d{8))$I^(d{11})S/; 
Var objExp=new RegExp(Expression); 
if(objExp.test(str)==trueX{ 
return true; 
yelse{ 
return false; 
} 


</script> 
当 用 户 单 击 头像 下 拉 列 表 选 择 头像 时 ， 显 示 头 像 图 片 。 程 序 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\08\login.php) 
<script language="javascript"> // 通 过 下 拉 列 表 选 择 头像 时 应 用 该 函数 
function showlogo(}{ 


document.images.img.src="images/tx/"+ 
document.form1.tx.options[document.form1 .tx.selectedindex].value; 


</script> 


<tr> 
<td> 选 择 头 像 : </td> 
<td><p><img src="images/tx/1.gif" id="img" name="img" width="60" height="60" /></p> 
<select size"1" id="tx" name="tx" onChange="showlogo()"> 
<option value="1.gif"> 头 像 1</option> 
<option value="2.gif"> 头 像 2</option> 
<option value="3.gif'> 头 像 3</option> 
<option value="4.gif"> 头 像 4</option> 
<option value="11.gif"> 头 像 11</option> 
<option value="6.gif'> 头 像 6</option> 
<option value="7.gif"> 头 像 7</option> 
.gif > 头像 8</option> 
.giP > 头像 9</option> 
<option value="10.gif'> 头 像 10</option> 
</select> 
当 用 户 单 击 “ 确 认 提 交 ” 按 钮 后 ， 将 表单 中 的 数据 提交 到 用 户 注册 信息 处 理 页 login_ok.php 中 ， 再 将 数 
据 添 加 到 数据 库 中 进行 存储 。 程 序 代码 如 下 : 
〈 代 码 位 置 : 光盘 \TMNVInstance\08\login_ok.php) 
<?php 
include("conn/conn.php"); 
ifisset($_POST[Submit]) and $_POST[Submit]==" 确 认 提 交 "){ 
S$username=$_POST[username']; 
$true_name=$_POST[true_name]; 
$password=$_POST[password']; 
$sex=$_POST['sex]; 
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Stel=$_POSTTtel]; 

$email=$_POST[email]; 

$qq=$_POSTTQQY; 

Sindexs=$_POST[indexs']; 

S$address=$_POST[address'; 

S$tx="images/tx/".$_POSTTtx"; 

if($_POST[password]==$_POST[password2]X{ 

Sinsert=mysql_query("insert into th_user(usemame,true_name,password,sex,tel,email,qq,indexs,address,tx) 

values('$username','$true_name','$password','$sex','$tel','$email','$qq','$indexs','$address",'$tx')", $conn); 


if($insert}{ 
echo "<script>alert(' 注 册 成 功 ! ');window.location.href='index.php';</script>;"; 
jelsef 
echo "<script>alert(' 注 册 失 败 ! ');window.location.href='index.php';</script>;"; 
} 
Jelse{ 
echo "<script>alert(' 两 次 输入 的 密码 不 一 致 ! ');window.location.href='login.php';</script>;"; 
} 
?> 


8.4 ”用 户 登 录 模块 设计 
茹 4 视频 讲解 : 光盘 \TM\Video\ 第 8 章 \ 用 户 登录 模块 设计 .exe 


8.4.1 用 户 登录 模块 概述 


用 户 登 录 模块 是 用 户 进 入 到 程序 系统 的 门户 ， 通 过 登录 模块 ， 可 以 对 登录 用 户 进行 身份 验证 ， 只 有 系 
统 的 合法 用 户 才 可 以 进入 系统 的 主 界面 。 整 个 登录 模块 的 实现 过 程 非常 简单 ， 相 信 读 者 会 很 快 掌握 。 登 录 
模块 的 运行 效果 如 图 8.16 所 示 。 


8.4.2 通过 JavaScript 脚本 判断 用 户 名 和 密码 是 否 为 空 


在 用 户 登 录 模块 中 ， 当 用 户 单 击 “ 登 录 ”按钮 时 , 使 用 JavaScript 脚本 技术 判断 用 户 名 和 密码 是 否 为 空 ， 
如 果 为 空 则 给 出 提示 。 具 体 代码 如 下 : 
<script language="javascript"> 
function check(){ // 自 定义 函数 
if(form3.user.value=="){ /| 判断 用 户 名 是 否 为 空 
alert(" 请 输入 用 户 名 "); 
form3.user.focus(); 
return false; 


上 
if(form3.pwd.value=="™")}{ /| 判断 密 码 是否 为 空 
alert(" 请 输入 密码 人"); 
form3.pwd.focus(); 
return false ; 


return true; 


</script> 
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8.4.3 ”用户 登录 模块 的 实现 过 程 


1. 页 面 设计 
用 户 登 录 模块 的 设计 非常 简单 ， 只 包括 两 个 表单 元 素 和 一 个 提交 按钮 ， 虽 然 内 容 少 ， 但 对 其 与 其 他 内 


容 进行 合理 的 搭配 也 是 非常 重要 的 ， 在 本 实例 中 设计 的 用 户 登 录 模块 的 效果 如 图 8.17 所 示 。 


区 好 得 录 明 日 科技 在 线 论坛 JiL0Llsnx1 5 FR 9: hme 属 司 夺 | 
a A 
图 8.16 用 户 登录 模块 的 运行 效果 图 8.17 用 户 登录 模块 的 效果 


用 户 登录 模块 的 页 面 设计 流程 如 下 。 

(1) 输出 系统 的 当前 时 间 ， 代 码 如 下 : 

<?php echo date("Y-m-d H:i:s");?> 

(2) 设计 用 户 登 录 添加 的 表单 元 素 和 按钮 。 

用 户 登 录 模块 中 涉及 到 的 HTML 表单 元 素 如 表 8.3 所 示 。 


表 8.3 用 户 登录 页 面 涉及 到 的 HTML 表单 的 重要 元 素 


2. 代码 设计 
用 户 登 录 模块 的 功能 主要 通过 两 个 步 又 来 完成 。 首 先 在 用 户 登录 页 面 中 输入 用 户 名 和 密码 ， 然 后 单 击 


“登录 ”按钮 ， 将 数据 提交 到 用 户 登 录 数 据 处 理 页 中 ， 对 提交 的 数据 进行 验证 ， 如 果 正 确 则 提示 用 户 登录 
成 功 ， 否 则 返回 到 用 户 登 录 页 面 。 其 中 用 户 登录 数据 处 理 页 中 的 代码 如 下 : 


(代码 位 置 ， 光盘 \TM\Instance\08\user.php) 


<?php 

session_start(); /| 初始 化 session 变量 
?> 

<?php 

include("conn/conn.php"); // 包 含 数 据 库 连 接 文件 


ifisset($_POST[user]) and isset($_POST[pwd])X{ 
$select=mysql_query("select * from tb_user where usemame=".$_POST[user]." and password=".$_POST[Tpwd].", 
$conn); 
if(mysql_num_rows($select)==1X{ 

$array=mysql_fetch_array($select); 

$_SESSION[user]=$_POST[user]; 

$_SESSION[sex]=$array[sex]; 

$_SESSION[email]=$array[email]; 

$_SESSION['qq]=$array['qq]; 

$_SESSION[tx]=$arrayftx]: 

echo "<script>alert(' 登 录 成 功 ! ');window.location.href='index.php'</script>;"; 
}else{ 

echo "<script>alert(' 登 录 失败 ! ');window.location.href='index.php'</script>;"; 
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8.5 ”帖子 分 类 管理 模块 设计 


8.5.1 帖子 分 类 管理 模块 概述 


本 在 线 论坛 是 针对 不 同 种 类 的 计算 机 编程 语言 开发 的 一 个 网 络 交流 平台 ， 所 以 在 本 论坛 中 根据 不 同 种 
类 的 计算 机 编程 语言 对 论坛 中 的 帖子 进行 分 类 管理 ， 帖 子 分 类 管理 模块 的 运行 效果 如 图 8.18 所 示 。 
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图 8.18 帖子 分 类 管理 模块 的 运行 效果 
8.5.2 使 用 SQL 语句 查询 数据 技术 


帖子 分 类 管理 功能 的 实现 主要 应 用 mysql_queryO0、mysql_num_rowsO 和 mysql_fetch_array0 函 数 ， 以 及 
SQL 语句 对 数据 库 中 的 数据 进行 查询 ， 然 后 根据 不 同 的 语言 种 类 进行 分 类 显示 。 下 面 对 应 用 的 函数 和 语句 
进行 详细 讲解 。 

1. mysql_query() 函 数 

返回 MySQL 结果 集中 一 个 单元 的 内 容 ， 经 常用 在 获取 少量 数据 的 程序 中 。 语 法 如 下 : 

mixed mysql_query ( resource result, int row [, mixed field] ) 

本 函数 获取 一 个 query 的 结果 。 参 数 field 可 以 是 字段 名 称 、 顺 序 或 者 是 fieldname.tablename 的 格式 。 如 
果 给 列 起 了 别名 (“select foo as bar from...”) ， 则 可 用 别名 蔡 代 列 名 。 

2. mysql_num_rows() 函 数 

使 用 mysql_num rows0 函 数 可 以 获取 由 select 语句 查询 到 的 结果 集中 行 的 数目 。 语 法 如 下 : 

int mysql_num_rows ( resource result ) 

此 命令 仅 对 SELECT 语句 有 效 。 要 取得 被 insert、update 或 者 delete 语句 所 影响 到 的 行 的 数目 ， 再 使 用 
mysql_ affected_rows( 〇 函数 。 

3. mysql_fetch_array() 函 数 

返回 根据 从 结果 集 获取 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false。 该 函数 在 获取 数据 库 中 数据 时 


是 非常 实用 的 。 语 法 如 下 : 


array mysql_fetch_array ( resource result [, int result_type] ) 
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mysql_fetch_array0 函 数 的 参数 说 明 如 表 8.4 所 示 。 
表 8.4 mysql_fetch_array() 函 数 的 参数 说 明 
参数 说 明 
result 资源 类 型 的 参数 ， 要 传 入 的 是 由 mysql_query0 函 数 返 回 的 数据 指针 


可 选项 ， 整 数 型 参数 ， 要 传 入 的 是 MYSQL ASSOC、MYSQL NUM、MYSQL BOTH 3 种 由 PHP 定义 
好 的 常数 之 一 ， 默 认 值 是 MYSQL BOTH 
result type | 用 MYSQL ASSOC 只 得 到 关联 索引 (相当 于 mysql_fetch_assoc0 函 数 ) 
用 MYSQL NUM 只 得 到 数字 索引 〈 相 当 于 mysql_fetch_ row0 函 数 ) 
用 MYSQL BOTH 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 


和 as 计 间 本 函数 返回 的 字段 名 是 区 分 大 小 写 的 。 


8.5.3 ”帖子 分 类 管理 模块 的 实现 过 程 


1. 页 面 设 计 

在 页 面 设计 中 ， 使 用 图 片 固然 可 以 使 页 面 变 得 非常 漂亮 ， 但 是 也 存在 一 些 弊端 ， 如 果 在 网 页 中 输出 大 
量 的 数据 时 使 用 图 片 作为 背景 ， 一 旦 处 理 不 好 ， 会 导致 数据 显示 与 背景 不 协调 ， 达 不 到 最 佳 效果 。 在 网 站 
中 使 用 简单 的 表格 来 完成 页 面 的 设计 就 不 会 出 现 上 述 问 题 ， 特 别 是 对 大 量 数据 输出 处 理 上 ， 使 用 表格 和 添 
加 背景 色 来 完成 页 面 设 计 是 非常 不 错 的 方法 ， 其 中 细 线 表格 边线 的 使 用 更 是 程序 员 设 计 网 页 的 首选 参数 。 
这 里 帖子 分 类 管理 模块 就 是 使 用 细 线 表格 来 完成 加 人 :加 
的 ， 效 果 如 图 8.19 所 示 。 


帖子 分 类 管理 模块 的 页 面 设计 流程 如 下 所 示 。 A 
(1) 使 用 includeO 语 句 包含 数据 库 服务 器 连接 图 8.19 帖子 分 类 管理 模块 的 效果 

文件 、 论 坛 的 头 文件 、 用 户 登录 模块 和 输出 登录 成 功用 户 的 用 户 名 文件 。 代 码 如 下 : 
<?php 


include("conn/conn.php"); 
include("index_01.php"); 
include("index_02.php"); 
include("index_03.php"); 
?> 


(2) 设计 一 个 细 线 表格 ， 对 数据 库 中 的 数据 进行 循环 输出 ， 这 里 按照 明日 科技 的 不 同类 别 的 图 书 进行 
帖子 的 分 类 管理 ， 其 中 包括 帖子 的 所 属 专区 、 表 情 图 和 版 主 、 创 建 日 期 、 专 区 帖子 的 主题 总 数 和 今日 发 布 
帖子 的 总 数 。 

细 线 表格 的 设计 方法 是 : 首先 创建 一 个 表格 ， 然 后 设置 表格 的 边框 、 填 充 和 间距 都 为 1， 接 着 设置 表格 
的 边框 颜色 为 白色 ， 背 景色 为 红色 ， 最 后 将 表格 中 单元 格 的 背景 色 设置 为 白色 ， 保 存 表格 。 这 时 再 浏览 这 
个 表格 时 ， 就 是 一 个 边线 为 红色 的 细 线 表格 。 

2. 代码 设计 

在 本 论坛 中 ， 对 帖子 进行 分 类 管理 主要 应 用 while 循环 语句 和 SQL 语句 ， 循 环 读 取 数 据 库 中 存储 的 有 
关 不 同 语言 书籍 的 数据 ， 并 对 数据 进行 分 页 显示 。 其 关键 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\08\index_04.php) 

<?php 

include("conn/conn.php"); 

if(isset($_GET['page])X{ 


® 
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Spage=$_GET[page'; 
Jelse{ 
$page=1; 


$page_count=3; 
$select=mysql_query("select * from tb_category",$conn); 
Srow=mysql_num_rows($select); 
$page_page=ceil($row/$page_count); 
$offect=($page-1)*$page_count' /获取 上 一 页 的 最 后 一 条 记录 ， 从 而 计算 下 一 页 的 起 始 记录 
$selects=mysql_query("select * from tb_category where id order by id desc limit $offect,$page_count", 
$conn); 
while($array=mysql_fetch_array($selects)){ 
$icon=substr($array['icon],3,30); 
> 
<table width="987" height="88" border="1" align="center" bordercolor="#FFCC99"> 
<tr> 
<td width="172" rowspan="2" align="center"><?php echo "<img src=\"S$icon\">";?></td> 
<td width="4113" height="28"><a href="Ib.php?category=<?php echo $array[f'category];?>"> 了 明日 科技 出 版 
的 [<?php echo $array[category];?>] 类 图 书 </a></td> 
<td width="340" rowspan="2"> 创 建 日 期 : <?php echo $array['create_date'];?><br> 
主题 总 数 : <?php $selectes=mysql_query("select * fom tb_content where category=".$array[category]",$conn); 
$count=mysql_num_rows($selectes); 
echo $count; 
?><br> 
今日 主题 数 : <?php $dates=date("Y-m-d"); 
Srows=mysql_query("select * from tb_content where release_date='$dates' and category=".$array 
[category]."",$conn); 
$counts=mysql_num_rows($rows); 
echo $counts;?></td> 


<tr> 
<td height="28" bgcolor="#FFCC66"> 版 主 : <?php echo $array['noderator]?></td> 
</tr> 
<?php 
} 
?> 
<tr> 
<td colspan="3"><div align="center"> 
<div align="right"> 共 <?php echo $page_page;?> 页 每 页 <?php echo $page_count;?> 条 当前 第 <?php 
echo $page; ?> 页 
<a href="index.php?page=1"> 首 页 </a> 
<a href="index.php?page=<?php if($page==1){echo $page=1; }else{ echo $page-1; }?>"> 上 一 页 </a> 
<a href="index.php?page=<?php if($page<$page_page){echo $page+1;}else{ echo $page_page;}?>"> 
下 一 页 </a> 
<a href="index.php?page=<?php echo $page_page; ?>"> 尾 页 </a></div></td> 
</tr> 
</table> 
当 单 击 图 书 类 别 超 链 接 时 ， 将 显示 该 类 别 的 所 有 帖子 信息 ， 并 进行 分 页 显示 。 其 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\08\ Ib.php) 
<?php 
include("conn/conn.php"); 
include("index_01.php"); 
include("index_02.php"); 
include("index_03.php"); 
?> 
<?php 
if(isset($_GET[page'])X{ 
$page=$_GET[page]; 
}else{ 
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$page=1; 


S$page_count=3; 

$select=mysql_query("select * from tb_content",$conn); 
$row=mysql_num_rows($select); 
S$page_page=ceil($row/$page_count); 


S$offect=($page-1)*$page_count; /| 获取 上 一 页 的 最 后 一 条 记录 ， 从 而 计算 下 一 页 的 起 始 记 录 

?> 

<table width="96%" border="1" align="center” cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" 
bgcolor="#E1DAEA"><tbody> 


<tr> 

<td height="30" colspan="2" align="center" bgcolor="#FFFBFO" class="STYLE4"> 
<?php 
if(isset($_GET['category])X{ 

S$category=$_GET['category']; 

echo urlencode($category); 
Jelse{ 

Scategory="™"; 


类 图 书 </td> 
<td width="2611" bgcolor="#FFFBFO">&nbsp;</td> 
</tr></tbody> 
</table> 
oe // 省 略 了 部 分 代码 


8.6 发 帖 模块 设计 
隔 视频 讲解 :光盘 \TM\Video\ 第 8 章 \ 发 帖 模块 设计 .exe 


8.6.1 ”发帖 模块 概述 


发 布 帖子 是 论坛 中 必 不 可 少 的 功能 ， 用 户 登 录 后 ， 单 击 “ 发 布 主 题 ” 超 链接 即 可 进入 发 帖 模 块 ， 发 帖 
模块 的 运行 效果 如 图 8.20 所 示 。 左 侧 显 示 发 帖 人 的 一 些 基本 信息 ， 右 侧 为 发 帖 区 域 ， 选 择 帖子 类 别 ， 输 入 
主题 、 内 容 ， 选 择 表情 后 ， 单 击 “ 主 题 提 交 ” 按 钮 即 可 发 布 帖子 。 


8.6.2 ”while 循环 语句 技术 


在 处 理发 帖 模 块 中 使 用 的 表情 图 时 ， 应 用 while 语句 循环 输出 表情 图 片 ， 在 读 取 数 据 库 中 存储 的 二 进 制 
格式 图 像 时 ， 应 用 了 mysql_query0 和 mysql_fetch_array0 函 数 。 使 用 $_SESSION 变量 输出 当前 用 户 信息 。 

使 用 while 语句 循环 输出 表情 图 片 的 代码 如 下 : 

<?php 

$select1=mysql_query("select * from tb_expression",$conn); 

while($array1=mysql_fetch_array($select1)){f 

?> 

<input type="radio" name="tx" value="<?php echo $array1[id];?>" /> <img src="<?php echo "image_1.php? 

recid=".$array1[id];?>" width="24" height="24"> 

<?php 

} 


?> 


CC 
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8.6.3 ”发 帖 模块 的 实现 过 程 


1. 页 面 设计 


发 帖 模块 的 设计 可 以 分 为 两 部 分 ， 第 一 部 分 是 输出 登录 用 户 的 个 人 信息 ， 第 二 部 分 是 发 帖 表 单 内 容 的 
设计 。 发 帖 模块 的 整体 设计 效果 如 图 8.21 所 示 。 
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图 8.20 发 帖 模块 的 运行 效果 图 8.21 发 帖 模块 的 整体 设计 效果 


发 帖 模 块 的 整体 设计 流程 如 下 。 
(1) 使 用 include0 语 名 获取 论坛 的 头 文件 和 登录 用 户 的 个 人 信息 。 代 码 如 下 : 
<?php 
include("index_01.php"); 
include("index_02.php"); 
?> 


(2) 设计 发 帖 模块 中 的 表单 元 素 。 


发 帖 页 面 涉及 到 的 HTML 表单 元 素 如 表 8.5 所 示 。 
表 8.5 发帖 页 面 涉及 到 的 HTML 表单 元 素 
名 称 | 类 型 含 义 重要 属性 
method="post" action="fbzt_ok.php" enctype="multipart/form-data" 

ed fm Se onSubmit="javascript:return fbzt_checkO:"> 
catego text 图 书 类 别 <?php echo $array['category']?>"><?php echo $array['category']:?> 
subject text 帖子 的 主题 名 称 
tx radio 表情 图 value="<?php echo $arrayl['id]:2>" 
content text 帖子 的 内 容 
Submit submit 主题 提交 
Submit2 TESet 重 置信 息 

2. 代码 设计 


发 帖 模块 主要 通过 一 个 文件 来 实现 ， 表 单 提交 及 数据 的 处 理 文件 是 fbzt.php 文件 。 

(1) 在 表单 提交 页 fbzt.php 中 ， 输 出 登录 用 户 的 个 人 信息 和 提交 的 表单 ， 以 及 论坛 中 使 用 的 表情 图 ， 
并 且 将 表单 中 的 数据 提交 到 fbzt.php 文件 中 ， 存 储 到 数据 库 中 。 

(2) 在 fbztphp 文件 中 ， 将 表单 提交 的 数据 存储 到 数据 库 中 。 其 关键 代码 如 下 : 


(代码 位 置 ， 光盘 \TM\Instance\08\fbzt.php) 
<?php 
include("conn/conn.php"); 
$select=mysql_query("select * from tb_category",$conn); 


全 
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ifisset($_SESSION[user]) and $_SESSIONF'user]!=null}{ 
ifisset($_POST[Submit]) and $_POST[Submit]==" 主 题 提交 "){ 
$date=date("Y-m-d"); 
Sinsert=mysql_query("insert into tb_content(category,subject,content,username,release_date) values 
(".$_POST[category].",".$_POST['subject].",".$_POST[content].",".$_SESSION['user].",'$date')",$conn); 
if($insert}{ 
echo "<script>alert(' 发 布 成 功 ! ');window.location.href='index.php';</script>;"; 
jelsef 
echo "<script>alert(' 发 布 失败 ! ');window.location.href='fbzt.php';</script>;"; 


} 
Jelse{ 
echo "<script>alert(' 请 先 登录 ! ');window.location.href='index.php';</script>"; 


> 


8.7 回帖 模块 设计 


敬 4 视频 讲解 : 光盘 \TM\Video\ 第 8 章 \ 回 帖 模块 设计 .exe 


8.7.1 回帖 模块 概述 


回帖 模块 主要 实现 的 是 对 用 户 提出 的 问题 进行 回复 ， 无 论 是 版 主 还 是 普通 用 户 都 可 以 对 提出 的 问题 发 
表 自 己 的 看 法 和 见解 。 回 帖 模块 的 运行 效果 如 图 8.22 所 示 。 


8.7.2 ” mysql 函数 处 理 技术 


其 中 应 用 到 的 MySQL 函数 包括 mysql_queryO 和 mysql_fetch_array0， 与 发 帖 模块 和 用 户 注册 模块 中 相 
同 ， 这 里 不 再 袭 述 。 


8.7.3 ”回帖 模块 的 实现 过 程 


1. 页 面 设计 

回帖 模块 的 设计 和 发 帖 模块 的 设计 相似 ， 该 模块 由 4 个 大 的 部 分 组 成 ， 第 1 部 分 是 输出 论坛 的 头 文件 
和 登录 者 的 信息 ， 第 2 部 分 是 输出 要 回复 的 帖子 的 发 起 人 信息 ， 第 3 部 分 是 回帖 表单 的 设计 ， 第 4 部 分 是 
尾 文件 的 设计 。 回 帖 模块 的 设计 效果 如 图 8.23 所 示 。 


Eel sel 


图 8.22 回帖 模块 的 运行 效果 图 8.23 ”回帖 模块 的 设计 效果 


第 8 章 合 实例 (一 ) 一 一 在 线 论坛 


可 帖 模块 的 整体 设计 流程 如 下 。 
(1) 使 用 include0 语 句 获取 论坛 的 头 文件 和 用 户 登录 信息 。 代 码 如 下 : 
<?php 

include("conn/conn.php"); 

include("index_01.php"); 

include("index_02.php"); 

Ws 


(2) 输出 帖子 发 起 人 的 信息 ， 包 括 用 户 名 、 图 片 、QQ 和 邮箱 等 ， 并 且 输 出 发 布 帖子 的 主题 。 
(3) 设计 回复 帖子 提交 的 表单 元 素 ， 包 括 主题 、 回 复 内 容 和 一 些 隐藏 的 选项 。 
回帖 页 面 涉 及 到 的 HTML 表单 元 素 如 表 8.6 所 示 。 


表 8.6 回帖 页 面 涉及 到 的 HTML 表单 元 素 


名 称 | 类 型 | 含 义 重要 属性 
method="post" action="fbzt_ok.php?subject=<?php echo $subject:?>" enctype= 
"multipart/form-data” 


Forml form 表单 


subject | text | 回复 主 题 | 
content 
| | 提交 | 


提交 
submit? | | 重 凡 | 
2. 代码 设计 
回帖 功能 的 实现 也 通过 一 个 文件 (hfztphp 文件 ) 来 完成 ， 该 文件 完成 帖子 回复 表单 的 设计 及 对 表单 提 
交 的 内 容 进行 处 理 。hfzt.php 文件 的 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\08\hfzt.php) 


<?php 

session_start(); /初始化 SESSION 变量 
if(isset($_SESSION['user])X{ // 淹 断 是 否 是 会 员 登录 
include("conn/conn.php"); /| 连接 数据 库 


include("index_01.php"); 
include("index_02.php"); 
ifisset($_POST[subject]) and $_POST[Submit]==" 提 交 "){ ” // 判 断 提交 数据 是 否 存在 
S$select=mysql_query("select * from tb_content where id=".$_GET[h_id]."",$conn); 
$array=mysql_fetch_array($select); 
S$category=$array['category']; 
$subject=$array['subject]; 
$date=date("Y-m-d"); 
Sinsert=mysql_query("insert into tb_resume_contents(resume_subject,resume_contents,resume_date, 
username,category,subject) values(".$_POST[subject].",".$_POST['content].",'$date',”".$_SESSION['user].", 
'$category','$subject')", $conn); 


if($insert}{ 
echo "<script>alert(' 回 复 成 功 ! ');window.location.href='|b.php';</script>"; 
}else{ 
echo "<script>alert(' 回 复 失败 ! ');window.location.href='"|b.php';</script>"; 
} 
了 
?> 
过 本 /省 略 了 部 分 代码 
<?php 
}else{ 
echo "<script>alert(' 您 没有 登录 ');window.location.href='index.php';</script>"; 
} 
?> 
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8.8 台 管 理 模块 设计 
贸 中 视频 讲解 : 光盘 \TM\Video\ 第 8 章 \ 后 台 管理 模块 设计 .exe 


8.8.1 后台 管理 模块 概述 


在 后 台 管理 模块 中 ， 主 要 实现 对 论坛 中 注册 用 户 、 发 布 的 帖子 、 回 复 帖子 和 非法 关键 字 进 行 管理 
台 管理 模块 主页 运行 效果 如 图 8.24 所 示 。 


8.8.2 ”URL 编码 和 SWITCH 框架 技术 


后 台 管 理 模 块 中 主要 应 用 的 函数 基本 已 经 在 前 台 的 内 容 中 详细 介绍 过 了 ， 这 里 介绍 前 面 没 有 提 到 的 
urlencode0 〇 函数 和 switch 语句 。 

1. urlencode() 函 数 

本 函数 对 指定 的 中 文字 符 串 进行 编码 , 返回 值 为 字符 串 类 型 。 urlencode0 函 数 将 传 入 的 字符 串 进 行 编码 。 
其 语法 如 下 : 

string urlencode ( string str ) 

参数 str 为 指定 要 编码 的 字符 串 。 


也 


pe 
使 用 urlencode0O 函 数 进行 编码 可 避免 该 问题 的 发 生 , 而 且 又 能 保证 传递 数据 的 安全 性 ， 可谓 是 一 举 两 得 。 
2. switch 语句 


通过 switch 语句 可 以 实现 一 个 简单 的 框架 网 页 的 效果 ， 这 种 框架 网 页 的 效果 在 论坛 的 后 台 管 理 中 是 非 
常 实用 的 。switch 语句 的 语法 如 下 : 


switch ( expr { /lexpr 条 件 为 变量 名 称 
case expr1: /icase 后 的 expr1 为 变量 的 值 
statement1; // 符 合 该 条 件 时 要 执行 的 部 分 

break ; /应 用 break 来 跳 离 循 环 体 

case expr2 : 

statement2 
default 

statementN 
break; 


} 

参数 expr 是 表达 式 的 值 ， 即 switch 语句 的 条 件 变量 的 名 称 ， 参 数 exprl 放置 于 case 语句 之 后 ， 是 要 与 
条 件 变量 expr 进行 匹配 的 值 中 的 一 个 ; statementl 是 在 参数 exprl 的 值 与 条 件 变量 expr 的 值 相 匹 配 时 执行 的 
代码 ; break 语句 实现 终止 语句 的 执行 ， 即 语句 在 执行 过 程 中 ， 遇 到 break 就 停止 执行 ， 跳 出 循环 体 ，default 
是 case 的 一 个 特例 ， 匹 配 了 任何 其 他 case 都 不 匹配 的 情况 ， 并 且 是 最 后 一 条 case 语句 。 


8.8.3 ”后台 主页 的 实现 过 程 


1. 页 面 设计 
后 台 管 理 模块 主页 应 用 的 是 简单 的 框架 效果 ， 通 过 switch 语句 来 完成 ， 可 以 分 为 3 个 部 分 。 后 台 管 理 


@_ 
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模块 的 设计 效果 如 图 8.25 所 示 。 


综合 


实例 (一 ) 一 一 在 线 论坛 


see 
a 


图 8.24 后 台 管理 系统 主页 的 运行 效果 
后 台 管 理 模块 主页 的 整体 设计 流程 如 下 。 


图 8.25 后 台 管理 模块 主页 的 设计 效果 


(1) 使 用 includeO 语 句 获取 论坛 的 头 文件 。 代 码 如 下 : 


<?php include("index_01.php");?> 
(2) 设计 后 台 管理 中 的 栏目 选项 ， 由 5 个 栏目 组 成 。 
(3) 根据 单 击 不 同 栏目 的 超 链接 输出 对 应 栏目 的 内 容 
<?php switch($pt){ 
case "栏目 管理 ": 


"主题 管理 ": 

include "ztgl.php"; 

break; 

case "回复 主题 管理 ": 
include "hfztgl.php"; 

break; 

case "用 户 管理 ": 
include "hygl.php"; 


break; 
case "危险 内 容 管理 ": 
include "ss.php"; 
break; 
default : 
include "Imgl.php"; 
break; 
. 


?> 


2. 代码 设计 


， 默 认输 出 栏目 管理 的 内 容 。 程 序 代码 如 下 : 


后 台 主 页 index.php 文件 ， 通 过 switch 语句 创建 网 页 框架 ， 实 现在 不 同 功能 模块 之 间 的 跳 转 操作 。 其 关 


键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\08\admin\index.php) 
<tr> 
<td> 


<a href="index.php?Imbs=<?php echo urlencode(" 栏 目 管理 ");?>"><img src="images/02_03.gif" 


alt="™ width="1711" height="211" border="0"></a></td> 
<ltr> 


<tr> 
<td> 


全 
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<a href="index.php?Imbs=<?php echo urlencode(" 主 题 管理 ");?>"><img src="images/02_06.gif" 
alt="™" width="1711" height="23" border="0"></a></td> 
</tr> 
<tr> 
<td> 
<a href="index.php?Imbs=<?php echo urlencode(" 回 复 主 题 管理 ");?>"><img src="images/ 
02_08.gif" alt=”" width="1711" height="23" border="0"></a></td> 
</tr> 
<tr> 
<td> 
<a href="index.php?Imbs=<?php echo urlencode(" 用 户 管理 ");?>"><img src="images/02_10.gif" 
alt="™" width="1711" height="24" border="0"></a></td> 
</tr> 
<tr> 
<td> 
<a href="index.php?Imbs=<?php echo urlencode(" 危 险 内 容 管 理 ");?>"><img src="images/ 
02_12.gif" alt=” width="1711" height="23" border="0"></a></td> 
</tr> 
本 /省略 了 部 分 代码 


8.8.4 栏目 管理 模块 的 实现 过 程 


在 后 台 管理 系统 中 主要 实现 5 个 功能 : 


功能 实现 的 方法 基本 都 是 相同 的 , 这 里 以 栏目 管理 为 例 = “gn 
进行 讲解 。 其 运行 效果 如 图 8.26 所 示 。 

栏目 管理 模块 由 Imgl.php 和 delete3.php 两 个 文件 
组 成 。 在 Imgl.php 文件 中 完成 栏目 管理 的 大 部 分 操作 ， 


可 以 将 其 分 解 为 3 部 分 内 容 。 
(1) 创建 栏目 添加 的 表单 , 完成 栏目 的 添加 操作 ， — 
并 且 将 数据 提交 到 本 页 , 在 本 页 中 将 数据 添加 到 指定 的 图 8.26 栏目 管理 模块 的 运行 效果 


数据 表 中 。 其 关键 代码 如 下 : 
<form action="index.php?Imbs= 栏目 管理 ”method="post” enctype="multipart/form-data"” name="myform" 
id="myform"> 
<tr> 
<td width="170" height="30" align="middle" bgcolor="#FFFBFO" class="style1">&nbsp; 版 主 : 
<input id="noderator" size="111" name="noderator /> 
</td> 
<td width="200" align="middle" bgcolor="#FFFBF0" class="style1"> 所 属 专区 : 
<select id="category" size="1" name="category"> 
<option value="asp" selected="selected">ASP</option> 
<option value="jsp">JSP</option> 
<option value="delphi">Delphi</option> 
<option value="visual basic">Visual Basic</option> 
<option value="visual foxpro">Visual Foxpro</option> 
<option value="visual c++">Visual C++</option> 
<option value="power">Power Buider</option> 
<option value=".net">.net</option> 
</select></td> 
<td align="left" valign="center" bgcolor="#FFFBF0" class="style1">&nbsp;&nbsp; 图 标 : 
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<input type="radio" name="icon" value="<?php echo S$top;?>" /> 
<img src="../images/tx/photo.jpg' width='411' height='40' /> 
<input type="radio" name="icon" value="<?php echo $df:?>" /> 
<img src="../images/tx/photoes.jpg' width='411' height="40' /></td> 
</tr> 
<tr> 
<td colspan="2" bgcolor="#FFFBFO">&nbsp;</td> 
<td bgcolor="#FFFBFO" class="style1">&nbsp; <input id="zhuijia”type="submit”value=" 追 加 栏目 " 
name="zhuijia" /></td> 
</tr> 
</form> 
<?php 
include("../conn/conn.php"); 
S$top=("../images/tx/photo.jpg"); 
$df=("../images/tx/photoes.jpg"); 
ifisset($_POST[noderator]) and $_POST[zhuijia]==" 追 加 栏目 "X 
$date=date("Y-m-d"); 
Sinsert=mysql_query("insert into tb_category(icon,category,noderator,create_date) values(".$_POST[icon'.", 
".$_POST['category].",".$_POST['noderator].",'$date')",$conn); 
if($insert}{ 
echo "<script>alert(' 追 加 成 功 ! ');window.location.href='index.php?Imbs=".$_GET[Imbs1].";</script>"; 
}else{ 
echo "<script>alert(' 追 加 失败 ! ');window.location.href='index.php?Imbs=".$_GET[Imbs1].";</script>"; 


} 
» 
2 
(2) 执行 查询 语句 ， 通 过 while 语句 完成 数据 库 中 存储 栏目 数据 的 分 页 输出 。 其 关键 代码 如 下 : 
<?php 


ifisset($_GET[page])) 
S$page=$_GET[page']; 
}else{ 
S$page=1; 
上 


$page_count=3; 

$select=mysql_query("select * from tb_category", $conn); 

$row=mysql_num_rows($select); 

S$page_page=ceil($row/$page_count); 

Soffect=($page-1)*$page_count; // 获 取 上 一 页 的 最 后 一 条 记录 ， 从 而 计算 下 一 页 的 起 始 记 录 

S$selects=mysql_query("select * from th_category where id order by id desc limit $offect,$page_count", $conn); 

if($selects\{ 

while($myrow=mysql_fetch_array($selects)){ ?> 
<tr class="style1" align="middle"> 

<td height="44" align="center" bgcolor="#FFFBFO"><img src="<?php echo $myrow['icon'];?>" width="40" 
height="40" /></td> 

<td height="44" align="center" bgcolor="#FFFBF0"><?php echo $myrow['category'];?></td> 

<td height="44" align="center" bgcolor="#FFFBF0O"><?php echo $myrow[noderator];?></td> 

<td height="44" align="center" bgcolor="#FFFBF0O"><?php echo $myrow['create_date'];?></td> 

<td height="44" align="center" bgcolor="#FFFBF0"><a href="delete3.php?Imbs=<?php echo urlencode(" 
栏目 管理 ");?>&amp;id=<?php echo $myrow[id1];?>"> 删 除 </a></td> 

</tr> 
<?php }?> 
(3) 创建 分 页 超 链 接 ， 实 现 不 同 页 面 之 间 的 跳 转 。 其 关键 代码 如 下 : 
<table width="80%" border="0" cellspacing="0" cellpadding="0"> 
<tr class="style4"> 
<td width="110%" class="#ff0000">&nbsp;&nbsp; 页 次 : <?php echo $page;?>/<?php echo $page_page;?> 


Ss 
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页 记录 : <?php echo $row;?> 条 &nbsp; </td> 
<td width="39%" class="#ff0000"> 

<a href="index.php?Imbs=<?php echo urlencode($_GET[Imbs"]);?>&amp; page=1"> 首 页 </a> 

<a href="index.php?Imbs=<?php echo urlencode($_GET[Imbs]);?>&amp; page=<?php if($page==1) 
{echo $page=1; }else{ echo $page-1; }?>"> 上 一 页 </a> 

<a href="index.php?Imbs=<?php echo urlencode($_GET[Imbs]);?>&amp; page=<?php if($page< 
$page_page){echo $page+1;}else{ echo $page_page;}?>"> 下 一 页 </a> 

<a href="index.php?Imbs=<?php echo urlencode($_GET[Imbs]);?>&amp; page=<?php echo 
$page_page; ?>"> 尾 页 </a></td> 

</tr> 


</table> 
(4) 删除 栏目 信息 ， 根 据 Imgl.php 页 面 传递 的 链接 ID， 在 delete3.php 文件 中 执行 栏目 的 删除 操作 。 
其 关键 代码 如 下 : 
<?php 
include("../conn/conn.php"); 
$delete=mysql_query("delete from tb_category where id=".$_GET[id]."",$conn); 
if($deleteX{ 
echo "<script>alert(' 删 除 成 功 ! ');window.location.href='index.php?Imbs=".$_GET[Imbs1].";</script>"; 


jelse{ 
echo "<script>alert( 删 除 失败 ! ');window.location.href='index.php?Imbs=".$_GET[Imbs1.";</script>"; 


8.9 小 结 


本 章 主要 介绍 了 在 线 论坛 的 基本 创建 流程 和 开发 思路 ， 以 及 各 个 功能 模块 的 实现 方法 。 论 坛 系统 的 实 
现 主要 把 握 3 个 方面 的 内 容 即 可 ， 即 如 何 向 数据 库 中 添加 数据 、 如 何 从 数据 库 中 读 取 数据 及 如 何 将 数据 库 
中 的 数据 删除 。 论 坛 中 所 有 功能 模块 的 实现 都 是 在 围绕 着 这 3 个 方面 进行 ， 用 户 注册 模块 、 发 帖 模块 、 回 
帖 模块 是 在 向 数据 库 中 添加 内 容 ， 而 查看 帖子 模块 、 帖 子 分 类 管理 模块 是 在 读 取 数据 库 中 的 数据 。 在 把 握 
住 这 3 个 大 方向 以 后 就 是 对 细节 进行 处 理 ， 可 以 根据 个 人 的 能 力 和 喜好 开发 出 各 式 各 样 的 论坛 系统 。 
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MySQL 数据 库 


( 铝 / 视频 讲解 : 25 分 钟 ) 


数据 库 作 为 程序 中 数据 的 主要 载体 ， 在 整个 项 目 中 扮演 着 重要 的 
角色 。PHP 自身 可 以 与 大 多 数 数 据 库 进行 连接 ， 但 MySQL 数据 库 是 
开源 界 所 公认 的 与 PHP 兼容 性 最 好 的 数据 库 。My5QL 具有 安全 性 、 
器 平台 性 、 体 积 小 和 高 效 等 将 点 ， 可 谓 PHP 的 “黄金 搭档 ”。My5QL 
操作 简便 、 易 学 易 用 ， 受 到 很 多 业界 人 士 的 青睐 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

WH 了 解 MySQL 数据 库 的 特点 

MW 掌握 MySQL5 的 特性 

Mi 掌握 如 何 连接 与 断 开 MySQL 数据 库 

MW 掌握 如 何 创建 、 删 除数 据 库 

MW 掌握 如 何 创建 、 修 改 和 删除 数据 表 

MH 掌握 如 何 管理 数据 表 

WI 掌握 phpMyAdmin 的 安装 配置 
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9.1 MySQL 简介 


PHP 在 开发 Web 站 点 或 一 些 管理 系统 时 , 需要 对 大 量 的 数据 进行 保存 。 XML 文件 和 文本 文件 虽然 可 以 
作为 数据 的 载体 ， 但 不 易 进行 管理 和 对 大 量 数据 的 存储 ， 所 以 在 项 目 开 发 时 ， 数 据 库 就 显得 非常 重要 。PHP 
可 以 连接 的 数据 库 种 类 较 多 ， 其 中 MySQL 数据 库 与 其 兼容 较 好 ， 在 PHP 数据 库 开 发 中 被 广泛 应 用 。 


9.1.1 什么 是 MySQL 


敬 4 视频 讲解 : 光盘 \TMNVVideo\ 第 9 章 \ 什 么 是 MySQL.exe 

MySQL 是 一 款 安全 、 跨 平台 、 高 效 的 ， 并 与 PHP、Java 等 主流 编程 语言 紧密 结合 的 数据 库 系 统 。 该 数 
据 库 系统 是 由 瑞典 的 MySQL AB 公司 开发 ,发 布 并 支持 ,由 MySQL 的 初始 开发 人 员 David Axmark 和 Michael 
Monty Widenius 于 1995 年 建立 的 。MySQL 的 象征 符号 是 一 只 名 为 Sakila 的 海豚 ,代表 着 MySQL 数据 库 的 
速度 、 能 力 、 精 确 和 优秀 本 质 。 

目前 MySQL 被 广泛 应 用 在 Internet 上 的 中 小 型 网 站 中 。 由 于 其 体积 小 、 速 度 快 、 总 体 拥有 成 本 低 ， 尤 
其 是 开放 源码 这 一 特点 ， 使 得 很 多 公司 都 采用 MySQL 数据 库 以 降低 成 本 。 

MySQL 数据 库 可 以 称 得 上 是 目前 运行 速度 最 快 的 SQL 语言 数据 库 之 一 。 除 了 具有 许多 其 他 数据 库 所 不 
具备 的 功能 外 ，MySQL 数据 库 还 是 一 种 完全 免费 的 产品 ， 用 户 可 以 直接 通过 网 络 下 载 MySQL 数据 库 ， 而 
不 必 支付 任何 费用 。 


9.1.2 MySQL 的 特点 


敬 i 视频 讲解 :光盘 \TM\Video\ 第 9 章 \MySQL 的 特点 .exe 

MySQL 具有 以 下 主要 特点 。 

功能 强大 : MySQL 中 提供 了 多 种 数据 库存 储 引擎 ， 其 各 有 所 长 ， 适 用 于 不 同 的 应 用 场合 ， 用 户 可 
以 选择 最 合适 的 引擎 以 得 到 最 高 性 能 ， 可 以 处 理 每 天 访问 量 超过 数 亿 的 高 强度 的 搜索 Web 站 点 。 
MySQL 5 支持 事务 、 视 图 、 存 储 过 程 、 触 发 器 等 。 

回 “支持 跨 平 台 : MySQL 支持 至 少 20 种 以 上 的 开发 平台 , 包括 Linux、Windows、 FreeBSD、IBMAIX、 
AIX、FreeBSD 等 。 这 使 得 在 任何 平台 下 编写 的 程序 都 可 以 进行 移植 ， 而 不 需要 对 程序 做 任何 的 修改 。 

运行 速度 快 : 高 速 是 MySQL 的 显著 特性 。 在 MySQL 中 ， 使 用 了 极 快 的 B 树 磁 盘 表 (MyISAM) 和 
索引 压缩 : 通过 使 用 优化 的 单 扫描 多 连接 ， 能 够 极 快 地 实现 连接 ， SQL 函数 使 用 高 度 优化 的 类 库 
实现 ， 运 行 速度 极 快 。 

加 ”支持 面向 对 象 ， PHP 支持 混合 编程 方式 。 编 程 方式 可 分 为 纯粹 面向 对 象 、 纯 粹 面向 过 程 、 面 向 对 
象 与 面向 过 程 混合 3 种 方式 。 

回 ”安全 性 高 : 灵活 和 安全 的 权限 与 密码 系统 ， 人 允许 基本 主机 的 验证 。 连 接 到 服务 器 时 ， 所 有 的 密码 
传输 均 采 用 加 密 形 式 ， 从 而 保证 了 密码 的 安全 。 

成 本 低 : MySQL 数据 库 是 一 种 完全 免费 的 产品 ， 用 户 可 以 直接 通过 网 络 下 载 。 

支持 各 种 开发 语言 : MySQL 为 各 种 流行 的 程序 设计 语言 提供 支持 , 为 它们 提供 了 很 多 的 API 函数 ， 
包括 PHP、ASPNET、Java、Eiffel、Python、Ruby、Tel、C、C++、Perl 语言 等 。 

数据 库存 储 容量 大 : MySQL 数据 库 的 最 大 有 效 表 尺寸 通常 是 由 操作 系统 对 文件 大 小 的 限制 决定 的 ， 
而 不 是 由 MySQL 内 部 限制 决定 的 。InnoDB 存储 引擎 将 mnoDB 表 保 存在 一 个 表 空 间 内 ， 该 表 空 间 
可 由 数 个 文件 创建 ， 表 空间 的 最 大 容量 为 64TB， 可 以 轻松 处 理 拥有 上 干 万 条 记录 的 大 型 数据 库 。 


加 
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支持 强大 的 内 置 函 数 : PHP 中 提供 了 大 量 内 置 函数 ， 几 平 涵盖 了 Web 应 用 开发 中 的 所 有 功能 。 它 内 
置 了 数据 库 连 接 、 文 件 上 传 等 功能 ，MySQL 支持 大 量 的 扩展 库 ， 如 MySQLi 等 ， 可 以 为 快速 开发 
Web 应 用 提供 便利 。 


9.1.3 ”MySQL 5 支持 的 特性 


MySQL 5 已 经 是 一 个 非常 成 熟 的 企业 级 应 用 的 数据 库 管理 系统 ， 在 许多 大 型 的 开源 项 目 中 被 广泛 地 应 
用 。MySQL 5 支持 许多 基本 和 高 级 的 特性 ， 例 如 : 


支持 各 种 数据 类 型 。 

回 “支持 事物 、 主 键 、 外 键 、 行 级 锁定 等 特性 。 

回 select 查询 语句 和 where 字句 中 ， 提 供 完 整 的 操作 符 和 函数 支持 。 
支持 子 查询 。 

支持 group by 和 orderby 子 句 。 

支持 各 种 聚合 函数 。 

支持 left outer join 和 right outer join 多 表 连 接 查 询 。 
支持 表 别名 、 字 段 别 名 。 

支持 跨 库 多 表 连 接 查 询 。 

支持 查询 缓存 ， 能 够 极 大 地 提升 查询 性 能 。 

支持 存储 过 程 、 视 图 和 触发 器 等 特性 。 
支持 多 平台 、 多 CPU 等 特性 。 

支持 嵌入 式 ， 可 以 将 MySQL 集成 到 嵌入 式 程序 中 。 


9.2 MySQL 下 载 


加 视频 讲解 : 光盘 \TM\Video\ 第 9 章 \MySQL 下 载 .exe 
MySQL 是 一 款 开 源 的 数据 库 软 件 ， 由 于 其 免费 特性 得 到 了 全 世界 用 户 的 喜爱 ， 是 目前 使 用 人 数 最 多 的 
数据 库 。 下 面 将 详细 讲解 如 何 下 载 该 数据 库 。 
(1) 打开 MySQL 主页 ， 地 址 是 http://www.mysql.com/， 如 图 9.1 所 示 。 
(2) 单 击 页 面 上 部 的 Downloads (GA) 链 接 ， 如 图 9.2 所 示 。 


MysQL 5.5 


图 9.1 MySQL 官方 网 站 图 9.2 单 击 下 载 页 面 链接 


@® 
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(3) 在 新 页 面 中 提供 了 MySQL、MySQL Workbench、ConnectorJ (MySQL JDBC 驱动 ) 的 下 载 ， 如 
9.3 所 示 。 

(4) 在 图 9.3 中 , 单 击 MySQL Community Server 5.5 链接 , 在 新 页 面 中 选择 Windows (x86, 32-bit), MSI 
Installer 选项 ， 单 击 Download 按钮 ， 如 图 9.4 所 示 。 


Windows (x86, 32-bit], MSI Installer 


(essa 


国 New Releases (GA) Windows (x86, 64-bit), MSI Installer 


MysQL communiy server ss 
(5.5,13 GA] 
Windows (x36, 32-bit), ZIP Archive 


Windows (x86, 64-bit), ZIP Archive 


图 9.3 常用 的 下 载 链 接 图 9.4 选择 下 载 的 MySQL 版 本 


(5) 在 新 页 面 中 单 击 No thanks, just take me to the downloads! 链 接 ， 跳 过 注册 步骤 ， 如 图 9.5 所 示 。 
(6) 在 新 页 面 中 选择 Asia 中 的 一 个 HTTP 链接 进行 下 载 ， 如 图 9.6 所 示 。 


HTP 
HTP Fr 
Hp Frm 


HP Fr 


No thanks, just thke me to the downloads! 


Hrm Frm 


图 9.5 直接 去 下 载 页 面 图 9.6 选择 下 载 的 链接 
(7) 对 下 载 的 文件 ， 选 择 执行 操作 如 图 9.7 所 示 。 
短 是 要 运行 述 是 异 存 来 旧 ftp 阐 .sdjp 的 mysql-5.513 win32msi (27.7 MB 运 FIR) Ms) “| | WAC | x 


9.7 选择 执行 的 操作 
(8) 下 载 进度 如 图 9.8 所 示 。 


mysql5513-wins2md 中 的 33% 。 于 和 2 分 折 49 依 sm | MMO | EEFBV 


9.8 下载 进度 提示 
(9) 下 载 完 成 如 图 9.9 所 示 。 


语 ”mral-5513-winizma 不 和 R 的 下 为 安 ,可 迪生 各 二 sot 入 WL sD) | MN -EFM 


9.9 下 载 完 成 提示 


py 
SS 虽然 IE9 浏览 器 提示 该 文件 可 能 有 危险 ， 可 以 单 击 右 侧 的 x 号 关闭 这 个 提示 信息 。 


® 
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9.3 MySQL 的 环境 安装 


MySQL 安装 文件 mysql-5.5.13-win32.msi 下 载 完 成 后 ， 下 面 讲解 其 安装 过 程 。 
(1) 双击 运行 下 载 后 的 程序 ， 如 图 9.10 所 示 。 
(2) 在 图 9.10 中 单 击 “ 运 行 ” 按 钮 ， 显 示 如 图 9.11 所 示 的 对 话 框 。 


Ee) 


Welcome to the MYSQL Server 5.5 Setup 
Whord 


Fe ese red rm sm ee, 
Oot rine sr re odPeseno 
无 涉 验 证 着 击 * 您 确定 要 运行 此 软件 四 We 


eR Dts 


ra 
ed 
Enlm | ee 


MySQL Ye ™ 


打开 收文 全 机 入 到 向) 

入 如 Ee 
9.10 询问 是 否 打开 文件 图 9.11 开始 运行 MySQL 安装 向 导 

(3) 在 图 9.11 中 单 击 Next 按钮 ， 显 示 如 图 9.12 所 示 的 对 话 框 。 

(4) 在 图 9.12 中 单 击 Next 按钮 ， 将 弹出 如 图 9.13 所 示 的 安装 类 型 选择 对 话 框 。 


基 MQL connr SS setup 
ndsw be 
wo ee pe tonnekeeme germant rth 


图 9.12 询问 是 否 接受 协议 图 9.13 选择 安装 类 型 


(5) 在 图 9.13 中 单 击 Typical 按钮 ， 显 示 如 图 9.14 所 示 的 对 话 框 。 
(6) 在 图 9.14 中 单 击 Install 按钮 开始 安装 ， 安 装 进度 如 图 9.15 所 示 。 


FTIR 


人 | 


Poe wat ehde se sono aaaratalsiysqtsave ss. 


Sue copmmanentes 
—「— 


9.14 ”确认 前 面 各 选择 步骤 的 对 话 框 图 9.15 MySQL 安装 进度 对 话 框 
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(7) 在 安装 完成 后 会 显示 如 图 9.16 和 图 9.17 两 个 广告 对 话 框 。 


MS emterprine 


| y A MSa Erterpae mnietin se mest 
comprehensve offeting of My5QL database 
ee ee 
MUSQL, we <tcn gee oy 
ee 
【enterprise “后 
1.The MySQL Enterprise Server.Themest miabhe mare sndp 
anal wats mor Pop ape 
ZS EnterprseMonto selee-M 


图 9.16 广告 对 话 框 1 


oe Si) ED [cw 


图 9.17 广告 对 话 框 2 


(8) 在 图 9.17 中 ， 单 击 Next 按钮 ， 显 示 如 图 9.18 所 示 的 对 话 框 。 
(9) 在 图 9.18 中 ， 选 中 Launch the MySQL Instance Configuration Wizard 复 选 框 ， 单 击 Finish 按钮 ， 显 


示 如 图 9.19 所 示 的 对 话 框 。 
PR Ee 


Completed the MySQL Server 5.5 Setup 
Wiaard 


Gd pe nen buton emt he sea Wasrs, 


图 9.18 安装 完成 对 话 框 


MYSQL Sorver Inctance Configuration Wizard Cy 
‘Welcome to the HySOL Server 
Confiourabon Weard 1.0.17.0 


unen Wemrd wil ow yeu te carmgum 
昌 rr an 和 cannot 
he 


图 9.19 开始 对 MySQL 数据 库 进行 配置 


(10) 在 图 9.19 中 ， 单 击 Next 按钮 ， 显 示 如 图 9.20 所 示 的 对 话 框 。 
(11) 在 图 9.20 中 ， 选 中 Detailed Configuration 〈 详 细 配 置 ) 单 选 按钮 ， 单 击 Next 按钮 ， 显 示 如 图 9.21 


所 示 的 对 话 框 。 


MyS QL Server inetance Configuration Wiaard 


MysQt Sever Instance Configuration 
| omoseme mar snes sener meane, 


Peae idlect acorfiouration yoe, 


5 Detalied corfuraton 


型 peere ms canguatontype to oeate rhe otmal ever 
prormmadhne 


CT Sondard Confouration 
Lethes only on acnes mat de net Wen heve 2 MS0E 
On 
ee 


MySQL Server instanee Configuration Wirard 


Wyse serve watonce Cornguabon 
Compigure ne MQ Severs Stemrer tanet 
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De re 
er 
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9.20 选择 使 用 哪 种 配置 方式 


图 9.21 选择 服务 器 类 型 


(12) 在 图 9.21 中 ， 选 中 Developer Machine (开发 者 机 器 ) 单 选 按钮 ， 单 击 Next 按钮 ， 如 图 9.22 


所 示 。 


(13) 在 图 9.22 中 ， 选 中 Multifunction Database〈 多 功能 数据 库 ) 单 选 按钮 ， 单 击 Next 按钮 ， 显 示 如 


_ 因 ) 
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9.23 所 示 的 对 话 框 。 


Wyse Server eranee cai Wed 


WD 


EH pom | 
see 
Rs er ms 


Suled for sinple wsb applications, rorileting et oggng 
二 onlatons tr Welles Mawes Pro Ondine 
anal 


| :ma | em oa | [EE co 


9.22 选择 数据 库 类 型 图 9.23 选择 InnoDB 表 空 间 保存 位 置 


(14) 在 图 9.23 中 ， 使 用 默认 设置 ， 单 击 Next 按钮 ， 显 示 如 图 9.24 所 示 的 对 话 框 。 
(15) 在 图 9.24 中 ， 使 用 默认 设置 ， 单 击 Next 按钮 ， 显 示 如 图 9.25 所 示 的 对 话 框 。 


CE] 基本 Te] 


qt ero et Config cion 


ontiguar the MSGL Sorom sm ndane 


east ret the metvotng spa 
(7 trable TCP/IP Networiong 


amenr NEE] Addtieme reaption fortnn po 


(7 oebhe Sirict Node 
De es option fo the vere to behave wore Me a rathionel 
onomene tomneaon, 了 ee 
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Ca ee | 


图 9.25 设置 端口 号 和 服务 器 SQL 模式 


图 9.24 选择 服务 器 并 发 访问 人 数 


Mn So ee 
不 要 修改 默认 的 端口 号 ， 除 非 3306 端口 已 经 被 占用 。 


(16) 在 图 9.25 中 ， 使 用 默认 设置 ， 单 击 Next 按钮 ， 显 示 如 图 9.26 所 示 的 对 话 框 。 


(17) 在 图 9.26 中 ,选中 Manual Selected Default Character Set / Collation 单 选 按钮 ， 设 置 字符 集 编码 为 
utf8， 单 击 Next 按钮 ， 显 示 如 图 9.27 所 示 的 对 话 框 。 


MS Server Inetanes Confguration Ward 
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图 9.26 设置 默认 的 字符 集 9.27 针对 Windows 系统 进行 的 设置 
(18) 在 图 9.27 中 ,选中 Install As Windows Service 和 Include Bin Directory in Windows PATH 复 选 框 ， 
单 击 Next 按钮 ， 显 示 如 图 9.28 所 示 的 对 话 框 。 
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MN 
.注意 在 安装 MySQL 数据 库 时 ， 一 定 要 牢记 在 上 述 步骤 中 设置 的 默认 用 户 root 的 密码 ， 这 是 我 们 在 
访问 MySQL 数据 库 时 必须 使 用 的 。 


(19) 在 图 9.28 中 输入 数据 库 的 密码 “111”， 单 击 Next 按钮 ， 显 示 如 图 9.29 所 示 的 对 话 框 。 


图 9.28 输入 数据 库 的 密码 图 9.29 确认 配置 对 话 框 
(20) 在 图 9.29 中 ， 单 击 Execute 按钮 ， 执 行 前 面 进行 的 各 项 配置 ， 如 图 9.30 所 示 。 在 配置 完成 后 ， 


如 图 9.31 所 示 。 


Em | 


9.30 ”显示 配置 进度 图 9.31 完成 配置 
到 此 MySQL 安装 成 功 , 如 果 要 查看 MySQL 的 安装 配置 信息 , 则 可 以 通过 MySQL 安装 目录 下 的 my.ini 
文件 来 完成 。 
在 my.ini 文件 中 , 可 以 查看 到 MySQL 服务 器 的 端口 号 、MySQL 在 本 机 的 安装 位 置 、MySQL 数据 库 文 
件 存储 的 位 置 ， 以 及 MySQL 数据 库 的 编码 等 配置 信息 。 参 考 内 容 如 图 9.32 所 示 。 


EE 


pe 时 11 be read by ths NySQL Ser 
Server correctly (see sbove) so it ze: 


图 9.32 my.ini 文件 的 配置 信息 
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9.4 启动、 连接 、 断 开 和 停止 MySQL 服务 器 


铬 il 视频 讲解 : 光盘 \TM\Video\ 第 9 章 \ 启 动 、 连 接 、 断 开 和 停止 MySQL 服务 器 .exe 

MySQL 数据 库 服务 器 的 安装 和 维护 相对 简便 ， 在 Linux/UNIX 操作 系统 中 ， 使 用 终端 命令 即 可 实现 对 
MySQL 数据 库 服 务 器 的 管理 , 而 在 Windows 操作 系统 中 , 不 仅 可 以 在 命令 提示 符 窗口 中 通过 命令 的 方式 管 
理 MySQL 服务 器 ， 还 可 以 通过 图 形 界面 进行 管理 。 下 面 将 以 Windows 操作 系统 为 例 介绍 如 何 对 其 进行 启 
动 、 关 闭 等 操作 。 


9.4.1 启动 MySQL 服务 器 


在 Windows 操作 系统 中 启动 MySQL 服务 器 的 方法 主要 有 两 种 : 通过 系统 服务 方式 和 在 命令 提示 符 下 
通过 命令 行 方式 。 

1. 通过 系统 服务 启动 MySQL 服务 器 

如 果 MySQL 设置 为 Windows 服务 , 则 可 以 通过 选择 “开始 /管理 工具 /5 服务 ”命令 , 打开 Windows 
服务 管理 器 ， 在 服务 器 的 列表 中 找到 mysql 服务 ， 单 击 鼠 标 右键 ， 在 弹出 的 快捷 菜单 中 选择 “启动 ”命令 ， 
启动 MySQL 服务 ， 如 图 9.33 所 示 。 

2. 在 命令 提示 符 下 启动 MySQL 服务 器 

选择 “开始 ”/“ 运 行 ” 命 令 ， 在 弹出 的 “运行 ”窗口 中 输入 “cmd” 命 令 ， 按 Enter 键 进入 命令 提示 符 
窗口 。 在 命令 提示 符 下 输入 : 

\> net start mysql 

按 Enter 键 后 ， 启 用 MySQL 服务 器 ， 如 图 9.34 所 示 。 
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图 9.33 通过 系统 服务 启动 MySQL 服务 器 图 9.34 在 命令 提示 符 下 启动 MySQL 服务 器 
9.4.2 ”连接 和 断 开 MySQL 服务 器 


对 MySQL 数据 库 进 行 管理 ， 首 先 要 建立 与 MySQL 服务 器 的 连接 。 下 面 分 别 介绍 连接 和 断 开 MySQL 
服务 器 的 方法 。 


® 
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1. 连接 MySQL 服务 器 

连接 MySQL 服务 器 通过 mysql 命令 实现 。 在 MySQL 服务 器 启动 后 ， 选 择 “开始 ”/“ 运 行 ”命令 ,在 
弹出 的 “运行 ”窗口 中 输入 “cmd” 命 令 ， 按 Enter 键 后 打开 命令 提示 符 窗口 。 在 命令 提示 符 下 输入 : 

\> mysql -h 主机 名 -u 用 户 名 -p 密码 

输入 完 命令 语句 后 ， 按 Enter 键 即 可 连接 MySQL 服务 器 ， 如 图 9.35 所 示 。 

2. 断 开 MySQL 服务 器 

连接 到 MySQL 服务 器 后 ， 可 以 通过 在 MySQL 提示 符 下 输入 “exit” 或 者 “quit” 命 令 断 开 MySQL 连 
接 ， 如 图 9.36 所 示 。 


了 


图 9.35 连接 MySQL 服务 器 | 图 9.36 断 开 MySQL 服务 器 
9.4.3 停止 MySQL 服务 器 


Bi 视频 讲解 : 光盘 \TM\Video\ 第 9 章 \ 停 止 MySQL 服务 器 .exe 

为 有 效 节 省 系统 资源 ， 在 使 用 完 MySQL 服务 后 需要 将 其 及 时 关闭 。 与 启动 MySQL 服务 类 似 ， 在 
Windows 操 中 ,停止 MySQL 服务 器 主要 有 两 种 方法 : 通过 服务 关闭 MySQL 服务 和 命令 提示 符 
下 通过 命令 行 的 方式 关闭 MySQL 服务 。 下 面具 体 介 绍 这 两 种 实现 方式 。 

1. 通过 系统 服务 停止 MySQL 服务 器 

如 果 使 用 的 是 安装 版 的 MySQL 数据 库 ， 则 可 以 通过 选择 “开始 ”/“ 管 理工 具 ”/“ 服 务 ”命令 ,打开 
Windows 服务 管理 器 ， 在 服务 器 的 列表 中 找到 mysql 服务 并 右 击 ， 在 弹出 的 快捷 菜单 中 选择 “停止 ”命令 ， 
停止 MySQL 服务 器 ， 如 图 9.37 所 示 。 
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应 用 这 3 种 方式 中 的 
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图 9.37 停止 MySQL 服务 器 
2. 在 命令 提示 符 下 停止 MySQL 服务 器 
选择 “开始 ”/“ 运 行 ”命令 ， 输 入 “cmd” 命 令 ， 进 入 命令 提示 符 窗口 ， 在 命令 提示 符 下 输入 : 
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\> net stop mysql 

按 Enter 键 即 可 停止 MySQL 服务 器 ， 如 图 9.38 所 示 。 

回 ”选择 “开始 ”/“ 运 行 ”命令 ， 输 入 “cmd” 命 令 ， 进 入 命令 提示 符 窗口 ， 在 命令 提示 符 下 输入 : 
\> mysqladmin -uroot shutdown -p111 

按 Enter 键 即 可 停止 MySQL 服务 ， 如 图 9.39 所 示 。 
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9.38 在 命令 提示 符 中 停止 MySQL 服务 器 图 9.39 使 用 mysqladmin 命令 停止 MySQL 服务 器 


9.5 phpMyAdmin 图 形 化 管理 工具 


phpMyAdmin 是 众多 MySQL 图 形 化 管理 工具 中 应 用 最 广泛 的 一 种 ， 是 一 款 使 用 PHP 开发 的 B/S 模式 
的 MySQL 客户 端 软件 ， 该 工具 是 基于 Web 跨 平台 的 管理 程序 ， 并 且 支 持 简 体 中 文 。 用 户 可 以 在 官方 网 站 
www.phpMyAdmin.net 上 免费 下 载 到 最 新 的 版 本 。 phpMyAdmin 为 Web 开发 人 员 提 供 了 类 似 于 Access、 SQL 
server 的 图 形 化 数据 库 操作 界面 ， 通 过 该 管理 工具 可 以 完全 对 MySQL 进行 操作 ， 例 如 ， 创 建 数据 库 、 数 据 
表 、 生 成 MySQL 数据 库 脚本 文件 等 。 

在 浏览 器 地 址 栏 中 输入 “http://localhost/phpMyAdmin/”， 再 在 弹出 的 对 话 框 中 输入 用 户 名 和 密码 ， 进 
入 phpMyAdmin 图 形 化 管理 主 界面 ， 接 下 来 就 可 以 进行 MySQL 数据 库 的 操作 。 下 面 将 分 别 介绍 如 何 创建 、 
修改 和 删除 数据 库 。 


9.5.1 数据 库 操作 管理 


1. 创建 数据 库 

在 phpMyAdmin 的 主 界面 ， 首 先 在 文本 框 中 输入 数据 库 的 名 称 db_study， 然 后 在 下 拉 列 表 框 中 选择 所 
要 使 用 的 编码 ， 一 般 选择 gb2312_Chinese_ci〔 简 体 中 文 编码 格式 ) ， 单 击 “ 创 建 ” 按 钮 ， 创 建 数 据 库 ， 如 
图 9.40 所 示 。 成 功 创建 数据 库 后 ， 将 显示 如 图 9.41 所 示 的 界面 。 
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图 9.40 phpMyAdmin 管理 主 界面 图 9.41 成 功 创建 数据 库 


2. 修改 数据 库 

在 如 图 9.41 所 示 的 界面 中 ， 在 右 侧 界面 还 可 以 对 当前 数据 库 进行 修改 。 单 击 界面 中 的 侈 操作 超 链 接 ， 
进入 修改 操作 页 面 。 

(1) 可 以 对 当前 数据 库 执行 创建 数据 表 的 操作 ， 只 要 在 创建 数据 表 的 提示 信息 下 面 的 两 个 文本 框 中 分 
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别 输入 要 创建 的 数据 表 的 名 称 和 字段 总 数 ， 然 后 单 击 “ 执 行 ” 按 钮 即 可 进入 到 创建 数据 表 结 构 页 面 。 

(2) 也 可 以 对 当前 的 数据 库 重 命名 ， 在 “重新 命名 数据 库 为 ”下 的 文本 框 中 输入 新 的 数据 库 名 称 ， 单 
击 “ 执 行 ”按钮 ， 即 可 成 功 修改 数据 库 名 称 。 修 改 数据 库 名 称 如 图 9.42 所 示 。 

3. 删除 数据 库 

要 删除 某 个 数据 库 ， 首 先 在 左 侧 的 下 拉 菜 单 中 选择 该 数据 库 ， 然 后 单 击 右 侧 界面 中 的 钢 副 除 超 链接 (如 
图 9.42 所 示 ) 即 可 成 功 删除 指定 的 数据 库 。 


9.5.2 ”管理 数据 表 


名 4 视频 讲解 : 光盘 \TM\Video\ 第 9 章 \ 管 理 数据 表 .exe 

管理 数据 表 是 以 选择 指定 的 数据 库 为 前 题 ， 然 后 在 该 数据 库 中 创建 并 管理 数据 表 。 下 面 就 来 介绍 如 何 
创建 、 修 改 和 删除 数据 表 。 

1. 创建 数据 表 

创建 数据 库 db_study 后 ， 在 右 侧 的 操作 页 面 中 输入 数据 表 的 名 称 和 字段 数 ， 然 后 单 击 “ 执 行 ” 按 钮 ， 
即 可 创建 数据 表 ， 如 图 9.43 所 示 。 
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图 9.42 ”修改 数据 库 图 9.43 ”创建 数据 表 


成 功 创建 数据 表 tb_admin 后 ， 将 显示 数据 表 结 构 界面 。 在 表单 中 对 各 个 字段 的 详细 信息 进行 输入 ， 
包括 字段 和 名、 数据 类 型 、 长 度 / 值 、 编 码 格式 、 是 否 为 空 、 主 键 等 ， 以 完成 对 表 结构 的 详细 设置 。 当 所 有 的 
信息 都 输入 后 ， 单 击 “ 保 存 ” 按 钮 ， 创 建 数据 表 结 构 ， 如 图 9.44 所 示 。 成 功 创建 数据 表 结构 后 ， 将 显示 如 
图 9.45 所 示 的 界面 。 
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图 9.44 创建 数据 表 结构 


虽 服务 器 :Iocalhost > 电 数据 库 : db_study 》 国 表 :tb_admin 
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图 9.45 成 功 创建 数据 表 
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of 
SR 单 击 “ 保 存 ” 按 钮 ， 可 以 对 数据 表 结构 以 横 版 显示 进行 表 结 构 编 辑 。 


2. 修改 数据 表 
一 个 新 的 数据 表 被 创建 后 ， 进 入 到 数据 表 页 面 中 ， 在 这 里 可 以 通过 改变 表 的 结构 来 修改 表 ， 可 以 执行 
添加 新 的 列 、 删 除 列 、 索 引 列 、 修 改 列 的 数据 类 型 或 者 字段 的 长 度 / 值 等 操作 ， 如 图 9.46 所 示 。 
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图 9.46 ”修改 数据 表 结 构 


3. 删除 数据 表 
要 删除 某 个 数据 表 ， 首 先 在 左 侧 的 下 拉 菜 单 中 选择 数据 库 ， 在 指定 的 数据 库 中 选择 要 删除 的 数据 表 ， 
然后 单 击 右 侧 界 面 中 的 锅 副 除 超 链接 〈 如 图 9.46 所 示 ) 即 可 成 功 删除 指定 的 数据 表 。 


9.5.3 ”管理 数据 记录 


菇 4 视频 讲解 : 光盘 \TM\Video\ 第 9 章 \ 管 理 数据 记录 .exe 

单 击 phpMyAdmin 主 界面 中 的 _ 膛 SQL 超 链接 ， 打 开 SQL 语句 编辑 区 。 在 编辑 区 输入 完整 的 SQL 语句 ， 
来 实现 数据 的 查询 、 添 加 、 修 改 和 删除 操作 。 

1. 使 用 SQL 语句 插入 数据 

在 SQL 语句 编辑 区 应 用 insert 语句 向 数据 表 tb_admin 中 插入 数据 后 ， 单 击 “ 执 行 ” 按 钮 ， 便 向 数据 表 
中 插入 了 一 条 数据 ,如 图 9.47 所 示 。 如 果 提 交 的 SQL 语句 有 错误 ， 系 统 会 给 出 一 个 警告 ， 提 示 用 户 修改 它 ; 
如 果 提 交 的 SQL 语句 正确 ， 则 弹出 如 图 9.48 所 示 的 提示 信息 。 


图 服务 器 : ocalhost ， 轧 数据 库 ; db_study ， 目 表 :也 -admin 
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图 9.47 使 用 SQL 语句 向 数据 表 中 插入 数据 图 9.48 成 功 添加 数据 信息 
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乓 人 昌 为 了 编写 方便 ， 可 以 利用 其 右 侧 的 属性 列表 来 选择 要 操作 的 列 ， 只 要 选中 要 添加 的 列 ， 双 击 其 
选项 或 者 单 击 “<<” 按 钮 添加 列 名称 。 
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2. 使 用 SQL 语句 修改 数据 

在 SQL 语句 编辑 区 应 用 update 语句 修改 数据 信息 ， 将 为 1 的 管理 员 的 名 称 改 为 纯净 水 ， 密 码 改 为 
111， 添 加 的 SQL 语句 如 图 9.49 所 示 。 

单 击 “ 执 行 ” 按 钮 ， 数 据 修改 成 功 。 比 较 修改 前 后 的 数据 如 图 9.50 所 示 。 
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图 9.49 添加 修改 数据 信息 的 SQL 语句 图 9.50 ”修改 单条 数据 的 实现 过 程 
3. 使 用 SQL 语句 查询 数据 
在 SQL 语句 编辑 区 应 用 select 语句 检索 指定 条 件 的 数据 信息 , 将 匡 小 于 4 的 管理 员 全 部 显示 出 来 , 添 


加 的 SQL 语句 如 图 9.51 所 示 。 
单 击 “ 执 行 ” 按 钮 ， 该 语句 的 实现 过 程 如 图 9.52 所 示 。 
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图 9.51 添加 查询 数据 信息 的 SQL 语句 9.52 查询 指定 条 件 的 实现 过 程 
除了 对 整个 表 的 简单 查询 外 ， 还 可 以 执行 复杂 的 条 件 查询 〈 使 用 where 子 句 提交 LIKE、ORDER BY、 
GROUP BY 等 条 件 查询 语句 ) 及 多 表 查 询 ， 读 者 可 通过 上 机 进行 实践 ， 灵 活 运用 SQL 语句 功能 。 
4. 使 用 SQL 语句 删除 数据 


在 SQL 语句 编辑 区 应 用 delete 语句 删除 指定 条 件 的 数据 或 全 部 数据 信息 ,删除 名 称 为 tm 的 管理 员 信息 ， 
添加 的 SQL 语句 如 图 9.53 所 示 。 


和 如 果 Delete 语 身后 面 没有 Where 条 件 值 ， 那 么 将 删除 指定 数据 表 中 的 全 部 数据 。 


单 击 “ 执 行 ”按钮 ， 弹 出 确认 删除 操作 对 话 框 ， 单 击 “确定 ”按钮 ， 执 行 数据 表 中 指定 条 件 的 删除 操 
作 。 该 语句 的 实现 过 程 如 图 9.54 所 示 。 
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9.53 ”添加 删除 指定 数据 信息 的 SQL 语句 9.54 删除 指定 条 件 的 实现 过 程 


5. 通过 form 表单 插入 数据 

选择 某 个 数据 表 后 ， 单 击 对 帮 入 超 链 接 ， 进 入 插入 数据 界面 ， 如 图 9.55 所 示 。 在 界面 中 输入 各 字段 值 ， 
单 击 “ 执 行 ” 按 钮 即 可 插入 记录 。 默 认 情 况 下 ， 一 次 可 以 插入 两 条 记录 。 

6. 浏览 数据 

选择 某 个 数据 表 后 ， 单 击 国 浏 览 超 链接 ， 进 入 浏览 界面 ， 如 图 9.56 所 示 。 单 击 每 行 记录 中 的 .成 按钮 ， 


_ 回 ) 
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可 以 对 该 记录 进行 编辑 ， 单 击 每 行 记录 中 的 兴 按钮 ， 可 以 删除 该 条 记录 。 
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图 9.56 浏览 数据 
7. 搜索 数据 
选择 某 个 数据 表 后 ， 单 击 忆 执 索 超 链接 ， 进 入 搜索 页 面 ， 如 图 9.57 所 示 。 在 这 个 页 面 中 ， 可 以 在 选择 
字段 的 列表 框 中 选择 一 个 或 多 个 列 ， 如 果 要 选择 多 个 列 ， 先 按 下 Ctrl 键 再 单 击 要 选择 的 字段 名 ， 查 询 结 果 
将 按照 选择 的 字段 名 进行 输出 。 


@ 方式 二 : 使 用 “ 按 例 查 询 "。 选择 要 查询 的 条 件 和 输入 要 查询 的 值 ， 单 击 “ 执 行 ”按钮 


图 9.57 搜索 查询 


在 该 界面 中 可 以 对 记录 按 条 件 进行 查询 。 查 询 方式 有 两 种 : 第 一 种 方式 选择 构建 where 语句 查询 。 直 接 


在 “where 语句 的 主体 ”文本 框 中 输入 查询 语句 ， 然 后 单 击 其 后 的 “执行 ”按钮 ; 第 二 种 方式 使 用 按 例 查询 
选择 查询 的 条 件 ， 并 在 文本 框 中 输入 要 查询 的 值 ， 单 击 “ 执 行 ”按钮 。 


@ 
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9.5.4 ”导入 导出 数据 


句 4 视频 讲解 : 光盘 \TM\Video\ 第 9 章 \ 导 入 导出 数据 .exe 
导入 和 导出 MySQL 数据 库 脚本 是 互 逆 的 两 个 操作 。 导 入 是 执行 扩展 名 为 .sql 文件 ， 将 数据 导入 到 数据 库 
中 ; 导出 是 将 数据 表 结构 、 表 记录 存储 为 .sql 的 脚本 文件 。 通过 导入 和 导出 的 操作 实现 数据 库 的 备份 和 还 原 。 


1. 导出 MySQL 数据 库 脚 本 
单 击 phpMyAdmin 主 界面 中 的 荆 导 出 超 链 接 , 打开 导出 编辑 区 ， 如 图 9.58 所 示 。 选择 导出 文件 的 格式 ， 
这 里 默认 使 用 选项 SQL， 选 中 “另存 为 文件 ” 复 选 框 ， 单 击 “ 执 行 ” 按 钮 ， 将 弹出 如 图 9.59 所 示 的 文件 下 
载 对 话 框 ， 单 击 “ 保 存 ” 按 钮 ， 将 脚本 文件 以 .sql 格式 存储 在 指定 位 置 。 


edo BD 
lum ns ms Em Sou mo Xes oom wom 
mer 

py oe. 
| 


Ee 
xeviewewe |© 数据 愉 生 成 针 本 文件 的 数据 表 
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el, © 


9.58 生成 MySQL 脚本 文件 设置 界面 图 9.59 存储 MySQL 脚本 对 话 框 


2. 导入 MySQL 数据 库 脚 本 
单 击 时 Import 超 链接 ,进入 执行 MySQL 数据 库 脚本 界面 , 单 击 “ 浏 览 "按钮 查找 脚本 文件 (如 db_study.sql) 
所 在 位 置 ， 如 图 9.60 所 示 ， 单 击 “ 执 行 ”按钮 ， 即 可 执行 MySQL 数据 库 脚本 文件 。 


“， 凶 服 务 器 :Iocalhost 局 获 基 库 : db_study 》 目 表 :由 TT 
_ 昌 浏览 三 结 构 “ 肤 sal 万 搜索 于: 括 入 [器 导出 扫 作 “| 蚀 评 空 因 阳 险 
Import 


文 二 文人 的 位 置 『 atadb study sq JD 各 (最大 梁 制 : 2048 KS) 


NF 
Imponied fle compression wil be automaticaNl detecled 和 @ 选择 mysql 数据 库 脚本 文件 


@ 单 击 “ 执 行 ”按钮 即 可 生效 


图 9.60 执行 MySQL 数据 库 脚 本 文件 
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人 os 寺 总 在 执行 MySQL 脚本 文件 前 , 首先 检测 是 否 有 与 所 导入 同名 的 数据 库 , 如 果 没 有 同名 的 数据 库 ， 

则 首先 要 在 数据 库 中 创建 一 个 名 称 与 数据 文件 中 的 数据 库 名 相同 的 数据 库 ， 然 后 再 执行 MySQL 数据 库 
脚本 文件 。 另 外 ， 在 当前 数据 库 中 ， 不 能 有 与 将 要 导入 数据 库 中 的 数据 表 重 名 的 数据 表 存 在 ， 如 果 有 重 
名 的 表 存 在 ， 导 入 文件 就 会 失败 ， 提 示 错 误 信 息 。 


入 明 读者 也 可 通过 单 击 phpMyAdmin 图 形 化 工具 左 侧 区 的 辆 按钮， 在 打开 的 对 话 框 中 ， 单 击 “ 导 
入 文件 ” 超 链 接 ， 然 后 选择 脚本 文件 所 在 的 位 置 ， 从 而 执行 脚本 文件 。 


9.5.5 ” phpMyAdmin 设置 编码 格式 


顽 4 视频 讲解 : 光盘 \TMVVideo\ 第 9 章 \phpMyAdmin 设置 编码 格式 .exe 

将 页 面 、 程 序 文件 、 数 据 库 与 数据 表 设 置 统一 的 编码 格式 可 以 使 程序 运行 时 不 至 于 出 现 乱 码 。 一 般 情 
况 下 , 设置 页 面 的 编码 格式 由 HTML 中 的 meta 标签 实现 , 设置 程序 文件 的 编码 格式 是 由 headerO 函 数 实 现 ， 
设置 数据 库 与 数据 表 的 编码 格式 可 以 通过 使 用 phpMyAdmin 实现 。 下 面 以 实例 详细 讲解 一 下 如 何 为 新 创建 
的 数据 库 设 置 编码 格式 。 具 体 步骤 如 下 ; 

(1) 登录 到 phpMyAdmin 图 形 化 工具 页 面 ， 创 建 数据 库 名 称 ， 并 为 新 创建 的 数据 库 选 择 编码 格式 ， 如 
图 9.61 所 示 。 

(2) 创建 数据 表 ， 定 义 数据 表 字 段 ， 并 为 新 创建 的 数据 表 设 置 编码 格式 ， 如 图 9.62 所 示 。 
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图 9.61 设置 数据 库 编码 格式 图 9.62 设置 字段 编码 格式 
9.5.6 ” ”phpMyAdmin 添加 服务 器 新 用 户 
区 4 视频 讲解 : 光盘 \TM\Video\ 第 9 章 \phpMyAdmin 添加 服务 器 新 用 户 .exe 


在 phpMyAdmin 图 形 化 管理 工具 中 , 不 但 可 以 对 MySQL 数据 库 进行 各 种 操作 , 而 且 可 以 添加 服务 器 新 
用 户 ， 并 对 新 添加 的 用 户 设置 权限 。 


@_ 


第 9 章 MySQL 数据 库 全 全 全 


在 phpMyAdmin 中 添加 MySQL 服务 器 新 用 户 的 步骤 如 下 : 

(1) 单 击 phpMyAdmin 主 界面 中 的 瞻 权 限 超 链 接 ， 打 开 服 务 器 用 户 操作 界面 ， 如 图 9.63 所 示 。 

(2) 在 该 界面 中 ， 单 击 “ 添 加 新 用 户 ” 按 钮 ， 进 入 到 如 图 9.64 所 示 界 面 ， 设 置 用 户 名 、 密 码 、 主 机 和 
新 用 户 的 权限 。 设 置 完成 后 ， 单 击 “ 执 行 ”按钮 ， 完 成 对 新 用 户 的 添加 操作 ， 返 回 主页 面 ， 将 提示 新 用 户 
添加 成 功 。 


EE 


ET 


TT TT 
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图 9.63 服务 器 用 户 一 览 表 图 9.64 设置 添加 用 户 信息 


9.5.7 phpMyAdmin 中 重 置 MySQL 服务 器 登录 密码 


顽 4 视频 讲解 :光盘 \TM\Video\ 第 9 章 \phpMyAdmin 中 重 置 MySQL 服务 器 登录 密码 .exe 
在 phpMyAdmin 图 形 化 管理 工具 中 , 不 但 可 以 对 MySQL 数据 库 进行 各 种 操作 , 而 且 可 以 对 用 户 的 权限 
进行 设置 ， 同 时 还 可 以 对 MySQL 服务 器 的 登录 密码 进行 重 置 。 
在 phpMyAdmin 中 重 置 MySQL 服务 器 登录 密码 的 步骤 如 下 : 
(1) 单 击 phpMyAdmin 主 界面 中 的 哆 权限 超 链接 ， 打 开 服 务 器 用 户 操作 界面 ， 如 图 9.65 所 示 。 
(2) 在 该 界面 中 ， 可 以 对 指定 用 户 的 权限 进行 编辑 、 可 以 添加 新 用 户 和 删除 指定 的 用 户 。 这 里 选择 指 
定 的 用 户 ， 单 击 贸 | (编辑 权限 ) 超 链接 ， 对 指定 用 户 的 权限 进行 设置 ， 进 入 到 如 图 9.66 所 示 的 界面 。 
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图 9.65 服务 器 用 户 一 览 表 图 9.66 编辑 用 户 权 限 


在 图 9.66 所 示 的 界面 中 ， 可 以 设置 用 户 的 权限 、 修 改 密码 、 更 改 登 录用 户 信息 和 复制 用 户 。 在 输入 新 
密码 和 确认 密码 后 ， 单 击 “ 执 行 ”按钮 ， 完 成 对 用 户 密码 的 修改 操作 ， 返 回 主页 面 ， 将 提示 密码 修改 成 功 。 
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9.6 小 结 


本 章 对 MySQL 数据 库 的 基本 概念 、MySQL 5 的 新 特性 进行 了 介绍 ， 并 详细 介绍 了 Windows 系统 下 使 
用 “命令 提示 符 ” 窗 口 创建 和 维护 MySQL 数据 库 和 数据 表 的 方法 ， 在 讲解 过 程 中 注重 实践 和 常用 命令 的 讲 
解 ， 一 切 以 能 够 帮助 读者 打 好 基础 为 出 发 点 。 通 过 本 章 的 学 习 ， 读 者 能 够 了 解 MySQL 数据 库 的 基本 操作 和 
维护 方法 ， 掌 握 MySQL 数据 库 中 最 基本 和 最 常用 命令 的 语法 格式 ， 并 能 够 具备 基本 管理 和 维护 MySQL 数 
据 库 的 能 力 。 


9.7 学 习 成 果 检 验 


1. 创建 一 个 数据 库 student， 并 在 student 中 创建 数据 表 stu。 完 成 表 的 创建 后 ， 向 数据 表 中 插入 两 条 记 
录 ， 然 后 删除 第 一 条 记录 ， 最 后 显示 表 结构 。 (答案 位 置 ， 光盘 \TM\Instance\09\9.1) 

2. 在 数据 库 student 中 ， 创 建 商品 信息 表 stu。 (答案 位 置 : 光盘 \TMNInstance\09\9.2) 

3. 创建 数据 库 stu 然后 将 stu 更 名 为 student， 并 确保 更 名 成 功 。 (答案 位 置 : 光盘 \TM\Instance\09\9.3) 


/4。 


MySQL 存储 引擎 与 运算 符 


( 名! 视频 讲解 : 33 分 钟 ) 


使 用 存储 引擎 可 以 加 快 查 询 的 速度 ， 并 且 每 一 种 引擎 都 存在 不 同 
的 含义 。MySQL 的 数据 类 型 是 数据 的 一 种 属性 ， 它 可 以 决定 数据 的 存 
储 格式 、 有 效 范围 和 相应 的 限制 。 并 且 可 以 让 读者 了 解 如 何 选 择 合适 
的 数据 类 型 ， 以 及 My5QL 中 讲述 的 运算 符 ， 其 作用 是 用 来 指明 对 操 
作 数 所 进行 的 运算 。 本 章 将 介绍 MySQL 的 存储 引擎 、 数 据 类 型 的 使 
用 ， 以 及 各 种 运算 符 的 详细 讲解 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

WI 了解 MySQL 存储 引擎 

WH 了 解 MySQL 数据 类 型 、 运 算 符 

Wm 掌握 算 运 算 符 的 使 用 方法 

MW 掌握 字符 诗 和 日 期 的 数据 类 型 

MW 熟悉 比较 运算 符 的 使 用 

”熟悉 逻辑 运算 符 的 使 用 
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10.1 MySQL 存储 引擎 


存储 引擎 其 实 就 是 如 何 存储 数据 、 如 何 为 存储 的 数据 建立 索引 和 如 何 更 新 、 查 询 数据 等 技术 的 实现 方 
法 。 因 为 在 关系 数据 库 中 数据 的 存储 是 以 表 的 形式 存储 的 ， 所 以 存储 引擎 也 可 以 称 为 表 类 型 〈 即 存储 和 操 
作 此 表 的 类 型 ) 。 在 Oracle 和 SQL Server 等 数据 库 中 只 有 一 种 存储 引擎 ， 所 有 数据 存储 管理 机 制 都 是 一 样 
的 。 而 MySQL 数据 库 提供 了 多 种 存储 引擎 。 用 户 可 以 根据 不 同 的 需求 为 数据 表 选 择 不 同 的 存储 引擎 ， 也 可 
以 根据 自己 的 需要 编写 自己 的 存储 引擎 。 


10.1.1 什么 是 MySQL 存储 引擎 


MySQL 中 的 数据 用 各 种 不 同 的 技术 存储 在 文件 〈 或 者 内 存 ) 中 。 这 些 技术 中 的 每 一 种 技术 都 使 用 不 同 
的 存储 机 制 、 索 引 技巧 、 锁 定 水 平 并 且 最 终 提供 广泛 的 不 同 的 功能 和 能 力 。 通 过 选择 不 同 的 技术 ， 你 能 够 
获得 额外 的 速度 或 者 功能 ， 从 而 改善 你 的 应 用 的 整体 功能 。 

这 些 不 同 的 技术 以 及 配套 的 相关 功能 在 MySQL 中 被 称 作 存储 引擎 〈 也 称 作 表 类 型 ) 。MySQL 默认 配 
置 了 许多 不 同 的 存储 引擎 ， 可 以 预先 设置 或 者 在 MySQL 服务 器 中 启用 。 你 可 以 选择 适用 于 服务 器 、 数 据 库 
和 表格 的 存储 引擎 ， 以 便 在 选择 如 何 存储 信息 、 如 何 检索 这 些 信 息 以 及 需要 的 数据 结合 什么 性 能 和 功能 时 
为 其 提供 最 大 的 灵活 性 。 


10.1.2 ”查询 MySQL 中 支持 的 存储 引擎 


区 4 视频 讲解 ， 光盘 \TM\Video\ 第 10 章 \ 查 询 MySQL 中 支持 的 存储 引擎 .exe 
使 用 SHOW ENGINES 语句 和 SHOW 语句 可 以 查询 MySQL 中 支持 的 存储 引擎 ， 下 面 分 别 使 用 这 两 个 
语句 来 查询 MySQL 中 支持 的 存储 引擎 。 
(1) 使 用 SHOW ENGINES 语句 查询 MySQL 中 支持 的 存储 引擎 。 其 查询 语句 如 下 : 
SHOW ENGINES; 
SHOW ENGINES 语句 可 以 用 “;” 结 束 ， 也 可 以 用 “\g” 或 者 “\G” 结 束 。“\g” 与 “;” 的 作用 是 相同 
的 ，“\G” 可 以 让 结果 显示 得 更 加 美观 。 使 用 SHOW ENGINES 语句 查询 的 查询 结果 如 图 10.1 所 示 。 
查询 结果 中 的 Engine 参数 指 的 是 存储 引擎 的 名 称 ; Support 参数 指 的 是 MySQL 是 否 支持 该 类 引擎 , YES 
表示 支持 ，Comment 参数 指 对 该 引擎 的 评论 。 
从 查询 结果 中 可 以 看 出 ，MySQL 支持 多 个 存储 引擎 ， 其 中 InnoDB 为 默认 存储 引擎 。 
(2) 使 用 SHOW 语句 查询 MySQL 中 支持 的 存储 引擎 。 其 查询 语句 如 下 : 
SHOW VARIABLES LIKE 'have%; 


使 用 SHOW VARIABLES 语句 查询 的 结果 如 图 10.2 所 示 。 

有 些 表 根 本 不 用 来 存储 长 期 数据 , 实际 上 用 户 需 要 完全 在 服务 器 的 RAM 或 特殊 的 临时 文件 中 创建 和 维 
护 这 些 数 据 ， 以 确保 高 性 能 ， 但 这 样 以 来 ， 也 存在 很 高 的 不 稳定 风险 。 还 有 一 些 表 只 是 为 了 简化 对 一 组 相 
同 表 的 维护 和 访问 ， 为 同时 与 所 有 这 些 表 交 互 提供 一 个 单一 接口 。 另 外 还 有 其 他 一 些 特别 用 途 的 表 ， 但 重 
点 是 : MySQL 支持 很 多 类 型 的 表 ， 每 种 类 型 都 有 自己 特定 的 作用 、 优 点 和 缺点 。MySQL 还 相应 地 提供 了 
很 多 不 同 的 存储 引擎 ， 可 以 以 最 适合 于 应 用 需求 的 方式 存储 数据 。MySQL 有 多 个 可 用 的 存储 引擎 ， 下 面 主 
要 介绍 InoDB、MYyYISAM 和 MEMEORY 3 种 存储 引擎 。 


@ 
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图 10.1 使 用 SHOW ENGINES 语句 查询 图 10.2 使 用 SHOW 语句 查询 
MySQL 中 支持 的 存储 引擎 MySQL 中 支持 的 存储 引擎 


10.1.3 InnoDB 存储 引擎 


InnoDB 已 经 开发 了 十 余年 ， 遵 循 CNU 通用 公开 许可 (GPL) 发 行 。 InnoDB 已 经 被 一 些 重量 级 因特网 
公司 所 采用 , 如 YYahoo!、Slashdot 和 Google, 为 用 户 操作 非常 大 -个 强大 的 解决 方案 .InnoDB 
给 MySQL 的 表 提供 了 事务 、 回 滚 、 崩 省 修复 能 力 和 多 版 本 并 发 挫 全 。 MysQL 从 3.23.34a 开始 
包含 InnoDB 存储 引擎 。InnoDB 是 MySQL 上 第 一 个 提供 外 键 约束 的 表 引 擎 ， 而 且 InnoDB 对 事务 处 理 的 能 
力 ， 也 是 MySQL 其 他 存储 引擎 所 无 法 与 之 比拟 的 。 下 面 介绍 InnoDB 存储 引擎 的 特 点 及 其 优 缺 点 。 

InnoDB 存储 引擎 中 支持 自动 增长 列 AUTO_INCREMENT。 自 动 增长 列 的 值 不 能 为 空 ， 且 值 必须 唯一 
MySQL 中 规定 自 增 列 必 须 为 主键 。 在 插入 值 时 ， 如 果 自 动 增长 列 不 输入 值 ， 则 插入 的 值 为 自动 增 长 后 的 值 ; 
如 果 输 入 的 值 为 0 或 空 (NULL)〉， 则 插入 的 值 也 为 自动 增长 后 的 值 ， 如 果 插 入 某 个 确定 的 值 ， 且 该 值 在 前 
面 没 有 出 现 过 ， 则 可 以 直接 插入 

InnoDB 存储 引擎 中 支持 外 键 (FOREIGN KEY) 。 外 键 所 在 的 表 为 子 表 ， 外 键 所 依赖 的 表 为 父 表 。 父 
表 中 被 子 表 外 键 关联 的 字段 必须 为 主键 . 当 删 除 、 更 新 父 表 的 某 条 信息 时 , 子 表 也 必须 有 相应 的 改变 ,InnoDB 
存储 引擎 中 ， 创 建 的 表 的 表 结 构 存 储 在 .frm 文件 中 。 数 据 和 索引 存储 在 innodb_data home _dir 和 
innodb_data_file_path 表 空 间 中 。 

InnoDB 存储 引擎 的 优势 在 于 提供 了 良好 的 事务 管理 、 崩 省 修复 能 力 和 并 发 控制 。 缺 点 是 其 读 写 效率 稍 
差 ， 占 用 的 数据 空间 相对 比较 大 。 

InnoDB 表 是 如 下 情况 的 理想 引擎 。 

回 ”更 新 密集 的 表 : InnoDB 存储 引擎 特别 适合 处 理 多 重 并 发 的 更 新 请 求 。 

回 事务， InnoDB 存储 引擎 是 唯一 支持 事务 的 标准 MySQL 存储 引擎 ， 这 是 管理 敏感 数据 (如 金融 信 

息 和 用 户 注册 信息 ) 的 必需 软件 。 

回 ”自动 灾难 恢复 : 与 其 他 存储 引擎 不 同 ,InnoDB 表 能 够 自动 从 灾难 中 恢复 。 虽然 MyISAM 表 是 能 在 

灾难 后 修复 ， 但 其 过 程 要 长 得 多 。 

Oracle 的 InnoDB 存储 引擎 广泛 应 用 于 基于 MySQL 的 Web、 电 子 商务 、 金 融 系 统 、 健 康 护理 以 及 零 
售 应 用 。 因 为 InnoDB 可 提供 高 效 的 ACID 独立 性 (Atomicity) 、 一 致 性 (Consistency)、 隔 离 性 (Isolation)〉、 
持久 性 (Durability) 兼容 事务 处 理 能 力 ， 以 及 独特 的 高 性 能 和 具有 可 扩展 性 的 构架 要 素 

另外 ，InnoDB 设计 用 于 事务 处 理应 用 ， 这 些 应 用 需要 处 理 崩 溃 恢 复 、 完整 性 、 高 级 别 的 用 户 并 发 
数 ， 以 及 响应 时 间 超 时 服务 水 平 合同 。 在 MySQL 5.5 中 ， 最 显著 的 增强 性 能 是 将 InnoDB 作为 默认 的 存储 
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引擎 。 在 MyISAM 以 及 其 他 表 类 型 依然 可 用 的 情况 下 ， 用 户 无 须 更 改 配置 ， 就 可 构建 基于 InnoDB 的 应 用 
程序 。 


10.1.4 ”MylSAM 存储 引擎 


MyISAM 存储 引擎 是 MySQL 中 常见 的 存储 引擎 ， 曾 是 MySQL 的 默认 存储 引擎 。 MyISAM 存储 引擎 是 
基于 ISAM 存储 引擎 发 展 起 来 的 ， 它 解决 了 ISAM 的 很 多 不 足 。MyISAM 增加 了 很 多 有 用 扩展 。 
1. MylSAM 存储 引擎 的 文件 类 型 
MyISAM 存储 引擎 的 表 存 储 成 3 个 文件 。 文 件 的 名 字 与 表 名 相同 。 扩 展 名 包括 fm、myd 和 myi。 
frm: 存储 表 的 结构 。 
myd: 存储 数据 ， 是 MYData 的 缩写 。 
myi: 存储 索引 ， 是 MYIndex 的 缩写 。 
2. MylSAM 存储 引擎 的 存储 格式 
基于 MyISAM 存储 引擎 的 表 支 持 3 种 不 同 的 存储 格式 ， 包 括 静 态 型 、 动 态 型 和 压缩 型 。 
(1) MyISAM 静态 
如 果 所 有 表 列 的 大 小 都 是 静态 的 ( 即 不 使 用 xBLOB、xTEXT 或 VARCHAR 数据 类 型 ) ，MySQL 就 会 
自动 使 用 静态 MyISAM 格式 。 使 用 这 种 类 型 的 表 性 能 非常 高 ， 因 为 在 维护 和 访问 以 预定 义 格式 存储 的 数据 
时 需要 很 低 的 开销 。 但 是 ， 这 项 优点 要 以 空间 为 代价 ， 因 为 每 列 都 需要 分 配 该 列 的 最 大 空间 ， 而 无 论 该 空 
间 是 否 真 正 地 使 用 。 
(2) MyISAM 动态 
如 果 有 表 列 (即使 只 有 一 列 ) 定 义 为 动态 的 (使 用 xBLOB、xTEXT 或 VARCHAR) ，MySQL 就 会 自 
动 使 用 动态 格式 。 虽 然 MyISAM 动态 表 占 用 的 空间 比 静 态 格式 所 占 空 间 少 ， 但 空间 的 节省 带 来 了 性 能 的 下 
降 。 如 果 某 个 字段 的 内 容 发 生 改 变 ， 其 位 置 很 可 能 就 需要 移动 ， 这 会 导致 碎片 的 产生 。 随 着 数据 集中 的 碎 
片 增加 ， 数 据 访 问 性 能 就 会 相应 降低 。 这 个 问题 有 两 种 修复 方法 : 
尽 可 能 使 用 静态 数据 类 型 。 
经 常 使 用 OPTIMIZE TABLE 语句 , 它 会 整理 表 的 碎片 , 恢复 由 于 表 更 新 和 删除 而 导致 的 空间 丢失 。 
(3) MyISAM 压缩 
有 时 会 创建 在 整个 应 用 程序 生命 周期 中 都 只 读 的 表 。 如 果 是 这 种 情况 ， 就 可 以 使 用 myisampack 工具 将 
其 转换 为 MYISAM 压缩 表 来 减少 空间 。 在 给 定 硬件 配置 下 (如 快速 的 处 理 器 和 低速 的 硬盘 驱动 器 ， 性 能 
的 提升 将 相当 显著 。 
3. MylSAM 存储 引擎 的 优 缺点 
MyISAM 存储 引擎 的 优势 在 于 占用 空间 小 ， 处 理 速度 快 。 缺 点 是 不 支持 事务 的 完整 性 和 并 发 性 。 


10.1.5 MEMORY 存储 引擎 


MEMORY 存储 引擎 是 MySQL 中 的 一 类 特殊 的 存储 引擎 。 其 使 用 存储 在 内 存 中 的 内 容 来 创建 表 ， 而 且 
所 有 数据 也 放 在 内 存 中 。 这 些 特性 都 与 mnoDB 存储 引擎 、MyISAM 存储 引擎 不 同 。 下 面 将 对 MEMORY 存 
储 引 擎 的 文件 存储 形式 、 索 引 类 型 、 存 储 周 期 和 优 缺 点 等 进行 讲解 。 

1. MEMORY 存储 引擎 的 文件 存储 形式 

每 个 基于 MEMORY 存储 引擎 的 表 实 际 对 应 一 个 磁盘 文件 。 该 文件 的 文件 名 与 表 名 相同 ， 类 型 为 frm。 


@ 
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该 文件 中 只 存储 表 的 结构 。 而 其 数据 文件 ， 都 是 存储 在 内 存 中 。 这 样 有 利于 对 数据 的 快速 处 理 ， 提 高 整个 
表 的 处 理 效 率 。 值 得 注意 的 是 ， 服 务 器 需要 有 足够 的 内 存 来 维持 MEMORY 存储 引擎 的 表 的 使 用 。 如 果 不 
需要 使 用 了 ， 可 以 释放 这 些 内 容 ， 甚 至 可 以 删除 不 需要 的 表 。 

2. MEMORY 存储 引擎 的 索引 类 型 

MEMORY 存储 引擎 默认 使 用 哈 希 (HASH) 索引 。 其 速度 要 比 使 用 B 型 树 (BTREE) 索引 快 。 如 果 读 
者 希望 使 用 B 型 树 索 引 ， 可 以 在 创建 索引 时 选择 使 用 。 

3. MEMORY 存储 引擎 的 存储 周期 

MEMORY 存储 引擎 通常 很 少 用 到 。 因 为 MEMORY 表 的 所 有 数据 是 存储 在 内 存 上 的 ， 如 果 内 存 出 现 异 
常 就 会 影响 到 数据 的 完整 性 。 如 果 重 启 机 器 或 者 关机 ， 表 中 的 所 有 数据 将 消失 。 因 此 ， 基 于 MEMORY 存 
储 引擎 的 表 生 命 周 期 很 得， 一 般 都 是 一 次 性 的 。 

4. MEMORY 存储 引擎 的 优 缺 点 

MEMORY 表 的 大 小 是 受到 限制 的 。 表 的 大 小 主要 取决 于 两 个 参数 , 分 别 是 max_rows 和 max_heap_table_ 
size。 其 中 ，max_rows 可 以 在 创建 表 时 指定 ， max_heap_table_size 的 大 小 默认 为 6MB， 可 以 按 需 要 进行 扩 
大 。 因 此 ， 其 存在 于 内 存 中 的 特性 ， 这 类 表 的 处 理 速度 非常 快 。 但 是 ， 其 数据 易 丢 失 ， 生 命 周 期 短 。 

创建 MySQL MEMORY 存储 引擎 的 出 发 点 是 速度 。 为 得 到 最 快 的 响应 时 间 ， 采 用 的 逻辑 存储 介质 是 系 
统 内 存 。 虽 然 在 内 存 中 存储 表 数 据 确实 会 提高 性 能 , 但 要 记 住 , 当 mysqld 守护 进程 朋 泪 时 , 所 有 的 MEMORY 
数据 都 会 丢失 。 

MEMORY 表 不 支持 VARCHAR、BLOB 和 TEXT 数据 类 型 ， 因 为 这 种 表 类 型 按 固定 长 度 的 记录 格式 存 
储 。 此 外 ， 如 果 使 用 版 本 4.1.0 之 前 的 MySQL， 则 不 支持 自动 增加 列 〈 通 过 AUTO_INCREMENT 属性 ) 。 
当然 ， 要 记 住 MEMORY 表 只 用 于 特殊 的 范围 ， 不 会 用 于 长 期 存储 数据 。 基 于 其 这 个 缺陷 ， 选 择 MEMORY 
存储 引擎 时 要 特别 小 心 。 

当 数据 有 如 下 情况 时 ， 可 以 考虑 使 用 MEMORY 表 。 

暂时 : 目标 数据 只 是 临时 需要 ， 在 其 生命 周期 中 必须 立即 可 用 。 

相对 无 关 : 存储 在 MEMORY 表 中 的 数据 如 果 突 然 丢 失 ， 不 会 对 应 用 服务 产生 实质 的 负面 影响 ， 

而 且 不 会 对 数据 完整 性 有 长 期 影响 。 

如 果 使 用 MySQL4.1 及 以 之 前 版 本 , MEMORY 的 搜索 比 MyISAM 表 的 搜索 效果 要 低 , 因为 MEMORY 
表 只 支持 散 列 索引 ， 这 需要 使 用 整个 键 进行 搜索 。 但 是 ，4.1 之 后 的 版 本 同时 支持 散 列 索引 和 B 树 索 引 。B 
树 索引 优 于 散 列 索引 的 是 ， 可 以 使 用 部 分 查询 和 通 配 查 询 ， 也 可 以 使 用 “<”、“>” 和 “>=” 等 操作 符 方 
便 数 据 挖掘 。 


10.1.6 ”如 何 选择 存储 引擎 


每 种 存储 引擎 都 有 各 自 的 优势 ， 不 能 笼统 地 说 谁 比 谁 更 好 ， 只 有 适合 不 适合 。 下 面 根据 其 不 同 的 特性 ， 
给 出 选择 存储 引擎 的 建议 。 
回 InnoDB 存储 引擎 : 用 于 事务 处 理应 用 程序 ， 具 有 众多 特性 ， 包 括 ACID 事务 支持 ， 支 持 外 键 。 同 
时 支持 崩溃 修复 能 力 和 并 发 控制 。 如 果 需 要 对 事务 的 完整 性 要 求 比较 高 ， 要 求实 现 并 发 控制 ， 那 
选择 InnoDB 存储 引擎 有 其 很 大 的 优势 。 如 果 需 要 频繁 地 进行 更 新 、 删 除 操作 的 数据 库 ， 也 可 以 选 
择 InnoDB 存储 引擎 。 因 为 ， 该 类 存储 引擎 可 以 实现 事务 的 提交 (Commit) 和 回 滚 (Rollback) 。 
MyISAM 存储 引擎 : 管理 非 事 务 表 ， 它 提供 高 速 存储 和 检索 ， 以 及 全 文 搜索 能 力 。MyISAM 存储 
引擎 插入 数据 快 ， 空 间 和 内 存 使 用 比较 低 。 如 果 表 主要 是 用 于 插入 新 记录 和 读 出 记录 ， 那 么 选择 
MyISAM 存储 引擎 能 实现 处 理 的 高 效率 。 如 果 应 用 的 完整 性 、 并 发 性 要 求 很 低 ， 也 可 以 选择 
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MyISAM 存储 引擎 。 


加 


MEMORY 存储 引擎 : MEMORY 存储 引擎 提供 “内 存 中 ” 表 ，MEMORY 存储 引擎 的 所 有 数据 都 


在 内 存 中 ， 数 据 的 处 理 速度 快 ， 但 安全 性 不 高 。 如 果 需 要 很 快 的 读 写 速度 ， 对 数据 的 安全 性 要 求 


较 低 ， 可 以 选择 MEMORY 存储 引擎 。MEMORY 存 
所 以 ， 这 类 数据 库 只 使 用 相对 较 小 的 数据 库 表 。 


储 引擎 对 表 的 大 小 有 要 求 ， 不 能 建 太 大 的 表 。 


以 上 存储 擎 引 的 选择 建议 是 根据 不 同 存储 引擎 的 特点 提出 的 ， 并 不 是 绝对 的 。 实 际 应 用 中 还 需要 根据 


各 自 的 实际 情况 进行 分 析 。 


10.1.7 ”设置 数据 表 的 存储 引擎 


铭 4 视频 讲解 : 光盘 \TM\Video\ 第 10 章 \ 设 置 数 据 表 的 存储 引擎 .exe 

下 面 创建 db_database03 数据 库 文件 ， 在 数据 库 中 创建 三 个 数据 表 ， 并 分 别 为 其 设置 不 同 的 存储 引擎 。 
以 此 来 诠释 这 三 种 不 同 存储 引擎 创建 的 数据 表 文件 有 什么 区 别 。 

(1) 创建 tb_001 数据 表 ， 设 置 存储 引擎 为 MyISAM， 生 成 的 数据 表 文 件 如 图 10.3 所 示 ， 由 3 个 不 同 


后 组 的 文件 组 成 。 


DIR ”| 


人 中 服务 器 :localhest ， 昂 数据 库 : db_database13 ) 表 :tb_001 
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10.3 ”创建 tb_001 数据 表 及 生成 的 数据 表 文 件 


(2) 创建 tb_002 数据 表 , 设置 存储 引擎 为 MEMORY, 生成 的 数据 表 文件 如 图 10.4 所 示 , 只 


| 
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10.4 创建 也 002 数据 表 及 生成 的 数据 表 文 件 
(3) 创建 tb_003 数据 表 ， 设 置 存储 引擎 为 mnoDB， 生 成 的 数据 表 文 件 如 图 10.5 所 示 ， 同 样 也 由 一 个 


后 缀 为 .frm 的 文件 组 成 。 
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遍 b 二 -二 Iacahex- 区 蔡 " 加 - 己 六 7- mn 2” 


二 因 服务 器 :localhost 、 区 孝 据 库 : db_databasef3 ， 目 表 :tb_003 
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铭 4 视频 讲解 : 光盘 \TM\Video\ 第 10 章 \MySQL 数据 类 型 .exe 


图 10.5 创建 也 003 数据 表 及 生成 的 数据 表 文 件 
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10.2 ”MySQL 数据 类 型 


在 MySQL 数据 库 中 ， 每 一 条 数据 都 有 其 数据 类 型 。MySQL 支持 的 数据 类 型 ， 主 要 分 成 3 类 : 数字 类 
型 、 字 符 串 〈 字 符 ) 类 型 、 日 期 和 时 间 类 型 。 


10.2.1 数字 类 型 


MySQL 支持 所 有 的 ANSIISO SQL 92 数字 类 型 。 这 些 类 型 包括 准确 数字 的 数据 类 型 (NUMERIC、 
DECIMAL 、INTEGER 和 SMALLINT) ， 还 包括 近似 数字 的 数据 类 型 (FLOAT 、REAL 和 DOUBLE 
PRECISION) 。 其 中 的 关键 字 INT 是 INTEGER 的 同义词 ， 关 键 字 DEC 是 DECIMAL 的 同义词 。 

数字 类 型 总 体 可 以 分 成 整 型 和 浮 点 型 两 类 ， 详 细 内 容 如 表 10.1 和 表 10.2 所 示 。 


表 10.1 整数 数据 类 型 


数据 类 型 取 值 范围 说 有明 单位 

TINYINT 符号 值 : -127 一 127 无 符号 值 : 0 一 255 最 小 的 整数 1 字 节 

BIT 符号 值 : -127 一 127 无 符号 值 : 0 一 255 最 小 的 整数 1 字 节 

BOOL 符号 值 : -127 一 127 无 符号 值 : 0 一 255 最 小 的 整数 1 字 节 

I 符号 值 : - 32768 一 32767 

sn 无 符号 值 : 0 一 65535 小 型 整数 2 
符号 值 : - 8388608 一 8388607 

站 无 符号 值 : 0 一 16777215 中 型 整数 3 
符号 值 : - 2147683648 一 2147683647 

a 无 符号 值 : 0~4294967295 标准 整数 3 
符号 值 : 

BIGINT -9223372036854775808 一 9223372036854775807 大 整数 8 字 节 
无 符号 值 : 0 一 18446744073709551615 
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表 10.2 浮 点 数据 类 型 
数据 类 型 取 值 范围 
FLOAT | + (-) 3.402823466E+38 
+ (-) 1.7976931348623157E+308 
+ (-) 2.2250738585072014E-308 


说 了 明 
单 精度 浮 点 数 
双 精 度 浮 点 数 


DOUBLE 


泌 


| 

(1) 选择 最 小 的 可 用 类 型 ， 如 果 值 永远 不 超过 127， 则 使 用 TINYINT 比 INT 强 。 
(2 ) 对 于 完全 都 是 数字 的 ， 可 以 选择 整数 类 型 。 

(3 ) 浮 点 类 型 用 于 可 能 具有 小 数 部 分 的 数 ， 如 货物 单价 、 网 上 购物 交付 金额 等 。 


10.2.2 ”字符 串 类 型 


字符 串 类 型 可 以 分 为 3 类 : 普通 的 文本 字符 串 类 型 (CHAR 和 VARCHAR)、 可 变 类 型 (TEXT 和 BLOB) 
和 特殊 类 型 (SET 和 ENUM) 。 它 们 之 间 都 有 一 定 的 区 别 ， 取 值 的 范围 不 同 ， 应 用 的 地 方 也 不 同 。 

(1) 普通 的 文本 字符 串 类 型 ， 即 CHAR 和 VARCHAR 类 型 ，CHAR 列 的 长 度 被 固定 为 创建 表 所 声明 
的 长 度 ， 取 值 在 1 一 255 之 间 ; VARCHAR 列 的 值 是 变 长 的 字符 串 ， 取 值 和 CHAR 一 样 。 下 面 介绍 普通 的 文 
本 字符 串 类 型 ， 如 表 10.3 所 示 。 


表 10.3 常规 字符 串 类 型 


类 型 取 值 范围 说 了 明 
固定 长 度 为 M 的 字符 串 ， 其 中 M 的 取 值 范围 为 0~~255。national 关键 字 


hotional 0_255 个 字符 | 指定 了 应 该 使 用 的 默认 字符 集 。binary 关键 字 指定 了 数据 是 否 区 分 大 小 写 

oe 55 个 学 符 | (默认 是 区 分 大 小 写 的) 。ASCI 关键 字 指定 了 在 改 列 中 使 用 latinl 字符 

ao 集 。unicode 关键 字 指 定 了 使 用 UCS 字符 集 

char 和 char(M) 类 似 

[national 

varchar(M) 长 度 可 变 ， 其 他 和 char(MD) 类 似 

bina 

(2) TEXT 和 BLOB 类 型 。 它 们 的 大 小 可 以 改变 ，TEXT 类 型 适合 存储 长 文本 ， 而 BLOB 类 型 适合 存 

储 二 进 制 数据 ， 支 持 任何 数据 ， 如 文本 、 声 音 和 图 像 等 。 下 面 介绍 TEXT 和 BLOB 类 型 ， 如 表 10.4 所 示 。 


表 10.4 TEXT 和 BLOB 类 型 


类 型 最 大 长 度 〈 字 节 数 ) 说 明 
TINYBLOB 2^8 一 1 (225) 小 BLOB 字段 
TINYTEXT 2^8 一 1 (225) 小 TEXT 字段 
BLOB 2^16~1 (65 535) 常规 BLOB 字段 
TEXT 2^16~1 (65 535) 常规 TEXT 字段 
MEDIUMBLOB 2^24~1 (16777215) 中 型 BLOB 字段 
MEDIUMTEXT 2^24~1 (16777215) 中 型 TEXT 字段 
LONGBLOB 2^32~1 (4 294 967 295) 长 BLOB 字段 


LONGTEXT 2^32 一 1 (4294 967 295) 长 TEXT 字段 
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(3) 特殊 类 型 SET 和 ENUM。 特 殊 类 型 SET 和 ENUM 的 介绍 如 表 10.5 所 示 。 
表 10.5 ENUM 和 SET 类 型 


该 类 型 的 列 只 可 以 容纳 所 列 值 之 一 或 为 NULL 
该 类 型 的 列 可 以 容纳 一 组 值 或 为 NULL 


Enum ("valuel", "value2". ...) 
Set ("valuel". "value2". ... 


| 


[a 


技巧 在 创建 表 时 ， 使 用 字符 囊 类 型 时 应 遵循 以 下 原则 
(1) 从 速度 方面 考虑 ， 要 选择 固定 的 列 ， 可 以 使 用 CHAR 类 型 。 
(2) 要 节省 空间 ， 使 用 动态 的 列 ， 可 以 使 用 VARCHAR 类 型 。 
(3) 要 将 列 中 的 内 容 限 制 在 一 种 选择 ， 可 以 使 用 ENUM 类 型 。 
(4) 允许 在 一 个 列 中 有 多 于 一 个 的 条 目 ， 可 以 使 用 SET 类 型 。 
(5) 如 果 要 搜索 的 内 容 不 区 分 大 小 写 ， 可 以 使 用 TEXT 类 型 。 
(6) 如 果 要 搜索 的 内 容 区 分 大 小 写 ， 可 以 使 用 BLOB 类 型 。 


10.2.3 ”日 期 和 时 间 数 据 类 型 


日 期 和 时 间 类 型 包括 DATETIME、DATE、TIMESTAMP、TIME 和 YEAR。 其 中 的 每 种 类 型 都 有 其 取 
值 的 范围 ， 如 赋予 它 一 个 不 合法 的 值 ， 将 会 被 “0” 代 蔡 。 下 面 介绍 日 期 和 时 间 数 据 类 型 ， 如 表 10.6 所 示 。 


表 10.6 “日 期 和 时 间 数 据 类 型 


类 型 取 值 范围 说 有 明 
DATE 日 期 ， 格式 YYYY-MM-DD 
TIME 时 间 ， 格 式 HH:MM:SS 


1000-01-01 00:00:00 
DATETIME 日 间 ， YYYY-MM-DD HH:MM:SS 
9999-12-31 23:59:59 a 


1970-01-01 00:00:00 WE 证 二 
TIMESTAMP 2037 年 的 某 个 时 间 时 间 标 签 ， 在 处 理 报告 时 使 用 显示 格式 取决 于 M 的 值 


YEAR 年 份 可 指定 两 位 数字 和 四 位 数字 的 格式 


在 MySQL 中 ， 日 期 的 顺序 是 按照 标准 的 ANSISQL 格式 进行 输出 的 。 


10.3 MySQL 运算 符 


数据 库 中 的 表 结 构 确 立 后 ， 表 中 的 数据 代表 的 意义 就 已 经 确定 。 而 通过 MySQL 运算 符 进行 运算 ， 就 可 
以 获取 到 表 结 构 以 外 的 另 一 种 数据 。 例如， 学 生 表 中 存在 一 个 birth 字段 ， 这 个 字段 是 表示 学 生 的 出 生年 份 。 
而 运用 MySQL 的 算术 运算 符 用 当前 的 年 份 减 学 生出 生 的 年 份 ， 那 么 得 到 的 就 是 这 个 学 生 的 实际 年 龄 数据 。 

这 就 是 MySQL 的 运算 符 ， 所 以 熟悉 并 掌握 运算 符 的 应 用 ， 对 于 操作 MySQL 数据 库 中 的 数据 是 非常 有 
用 的 。 下 面 就 来 熟悉 一 下 MySQL 支持 的 4 种 运算 符 都 具备 哪些 功能 。 

算术 运算 符 : 执行 算术 运算 ， 如 加 、 减 、 乘 、 除 等 。 

比较 运算 符 : 包括 大 于 、 小 于 、 等 于 或 者 不 等 于 等 。 主 要 用 于 数值 的 比较 、 字 符 串 的 匹配 等 方面 。 例 


全 
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如 ，LIKE、IN、BETWEEN AND 和 了 IS NULL 等 都 是 比较 运算 符 ， 还 包括 正则 表达 式 的 REGEXP 也 是 比较 
运算 符 。 

逻辑 运算 符 : 包括 与 、 或 、 非 和 异 或 等 逻辑 运算 。 其 返回 值 为 布尔 型 ， 真 值 (1 或 tue) 和 假 值 (0 或 
false) 。 

位 运算 符 : 包括 按 位 与 、 按 位 或 、 按 位 取 反 、 按 位 异 或 、 按 位 左 移 和 按 位 右 移 等 位 运算 。 注 意 位 运算 
必须 先 将 数据 转换 为 二 进 制 ， 然 后 在 二 进 制 格式 下 进行 操作 。 
SC 全 曙 

Sa 吉 辑 运算 符 和 位 运算 符 都 有 与 、 或 和 异 或 等 操作 。 但 是 ， 位 运算 符 必须 先 把 数值 变 成 二 进 制 类 
型 ， 然 后 再 进行 按 位 操作 。 运 算 完 成 后 ， 将 二 进 制 的 值 转换 为 原来 的 类 型 ， 返 回 给 用 户 。 还 辑 运算 直接 
进行 运算 ， 结 果 只 返回 真 值 (1 或 true) 和 假 值 (0 或 false )。 


10.3.1 算术 运算 符 


视频 讲解 : 光盘 \TM\Video\ 第 10 章 \ 算 术 运 算 符 .exe 
算术 运算 符 是 MySQL 中 最 常用 的 一 类 运算 符 。MySQL 支持 的 算术 运算 符 包括 加 、 减 、 乘 、 除 和 求 余 。 
下 面 列 出 算术 运算 符 的 符号 、 人 作用、 表达 式 的 形式 ， 如 表 10.7 所 示 。 


表 10.7 算术 运算 符 
符号 作 用 
+ 加 法 运算 
- 减法 运算 
乘法 运算 
/ 除法 运算 
% 求 余 运算 
DIV 除法 运算 ， 返 回 商 。 同 “/” 
MOD 求 余 运算 ， 返 回 余数 。 同 “%” 


a (+) 、 减 (-) 和 来 (*) 可 以 同时 运算 多 个 操作 数 。 除 号 (/) 和 求 余 运算 符 (%) 也 可 以 
同时 计算 多 个 操作 数 ， 但 是 这 两 个 符号 计算 多 个 操作 数 不 太 好 .DIV 和 MOD 这 两 个 运算 符 只 有 两 个 参 
数 。 进 行 除法 和 求 余 的 运算 时 ， 如 果 x2 参数 是 0 时， 计算 结果 将 是 空 值 (NULL ) 。 


例 10.1 使 用 算术 运算 符 对 tb_bookl 表 中 row 字段 值 进 
行 加 、 减 、 乘 、 除 运算 ， 计 算 结 果 如 图 10.6 所 示 。 (实例 位 
置 : 光盘 \TM\Instance\10\10.1) 

结果 输出 了 row 字段 的 原 值 ， 以 及 执行 算术 运算 符 后 得 
到 的 值 。 


10.3.2 ”比较 运算 符 


图 10.6 使 用 算术 运算 符 计算 数据 


区 9 视频 讲解 :光盘 \TM\Video\ 第 10 章 \ 比 较 运算 符 .exe 
比较 运算 符 是 查询 数据 时 最 常用 的 一 类 运算 符 。 SELECT 语句 中 的 条 件 语 句 经 常 要 使 用 比较 运算 符 。 通 


人 的 
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过 这 些 比较 运算 符 , 可 以 判断 表 中 的 哪些 记录 是 符合 条 件 的 。 比 较 运 算 符 的 符号 、 名 称 和 应 用 示例 如 表 10.8 
所 示 。 


表 10.8 比较 运算 符 


示 例 运算 符 名 称 示 例 
等 于 | Id=5 Is not null ma | Idis not null 
> Fu | Id>5 Between ma | Id between] and 15 
< 小 于 | Id<5 In ma | Id in (3.4.5) 
a | 1d=>5 Not in n/a | Name not in (shi.li) 
<= 小 于 等 于 | Id<=5 模式 匹配 | Name like (shi%%') 
!= 或 二 不 等 于 | Id!=5 模式 匹配 | Name not like ('shi%') 
Is null Idis null 常规 表达 式 Name 正则 表达 式 
下 面 对 几 种 较 常 用 的 比较 运算 符 进行 详解 。 


运算 符 “=” 
“=” 用 来 判断 数字 、 字 符 串 和 表达 式 等 是 否 相 等 。 如 果 相 等 ， 返 回 1， 否 则 返回 0。 
Ww A 
说明 在 运用 “=” 运 算 符 判 断 两 个 字符 是 否 相 同时 ， 数 据 库 系统 都 是 根据 字符 的 ASCII 码 进行 判断 
的 。 如 果 ASCII 码 相等 ， 则 表示 这 两 个 字符 相同 。 如 果 ASCII 码 不 相等 ， 则 表示 两 个 字符 不 同 。 忌 空 值 
(NULL ) 不 能 使 用 “=” 来 判断 。 


例 10.2 运用 “=” 运 算 符 查询 出 id 等 于 27 的 记录 ， 查 询 结果 如 图 10.7 所 示 。 (实例 位 置 ， 光 盘 \ 
TM\Instance\10\10.2) 

从 结果 中 可 以 看 出 ，id 等 于 27 的 记录 返回 值 为 1，id 不 等 于 27 的 记录 ， 返 回 值 则 为 0。 

运算 符 “<>” 和 “1=” 

“<>” 和 “!=” 用 来 判断 数字 、 字 符 串 、 表 达 式 等 不 相等 。 如 果 不 相等 ， 则 返回 1， 和 否则 ， 返 回 0。 
这 两 个 符号 也 不 能 用 来 判断 空 值 (NULL) 。 

例 10.3 运用 “<>” 和 “!=” 运 算 符 判 断 tb_book 表 中 row 字段 值 是 否 等 于 1、41、24。 运 算 结 果 如 
图 10.8 所 示 。 (实例 位 置 ， 光盘 \TM\Instance\10\10.3) 


图 10.7 使 用 “=” 查 询 记录 图 10.8 使 用 “一 ”和 “!=” 运 算 符 判断 数据 

结果 显示 返回 值 都 为 1， 这 表示 记录 中 的 row 字段 值 不 等 于 1、41、24。 

3. 运算 符 “>” 

“> ?用 来 判断 左边 的 操作 数 是 否 大 于 右边 的 操作 数 。 如 果 大 于 , 返回 1; 否则 , 返回 0。 同样 空 值 (NULL) 


不 能 使 用 “>” 来 判断 。 
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例 10.4 使 用 “>” 运 算 符 来 判断 tb_book 表 中 row 字段 值 是 否 大 于 90， 是 则 返回 1， 否 则 返回 0， 空 
值 返回 NULL。 运 算 结 果 如 图 10.9 所 示 。 (实例 位 置 : 光盘 \TM\Instance\10\10.4) 


/. 
说 日 
“说明 “<”、 “> 运算 符 都 与 “>” 运算 符 如 出 一 驾 ， 其 使 用 方法 基本 相同 ， 这 里 不 再 
其 过 了 . 


4. 运算 符 “IS NULL” 

“IS NULL” 用 来 判断 操作 数 是 否 为 空 值 (NULL) 。 操 作 数 为 NULL 时 ， 结 果 返 回 1; 否则 ， 返 回 0。 
IS NOT NULL 刚好 与 IS NULL 相反 。 

例 10.5 运用 “IS NULL” 运 算 符 来 判断 tb_book 表 中 row 字段 值 是 否 为 空 值 ， 查 询 结 果 如 图 10.10 所 
示 。 (实例 位 置 ， 光盘 \TM\Instance\10\10.5) 


图 10.9 使 用 “>” 运 算 符 查询 数据 图 10.10 使 用 “ISNULL” 运 算 符 来 判断 字段 值 是 否 为 空 
结果 显示 ，row 字段 值 为 空 的 返回 值 为 1， 不 为 空 的 返回 值 为 0。 


人 ey A 空 值 (NULL ) 。 
一 旦 使 用 ， 结果 将 返回 NULL。 如 果 要 判断 一 个 值 是 否 为 空 值 ， 可 以 使 用 “ 、 “IS NULL” 和 “IS 
NOT NULL” 来 判断 。 注 意 : NULL 和 ‘NULL” 是 不 同 的 ， 人 后 者 表示 一 个 由 4 个 字 


母 组 成 的 字符 串 。 


5. 运算 符 “BETWEEN AND” 

“BETWEEN AND ”用 于 判断 数据 是 否 在 某 个 取 值 范围 内 。 其 表达 式 如 下 : 

x1 BETWEEN m ANDn 

如 果 xl 大 于 等 于 m， 且 小 于 等 于 n， 结 果 将 返回 1， 否 则 将 返回 0。 

例 10.6 运用 “BETWEEN AND” 运 算 符 判断 tb_book 表 中 ，row 字段 的 值 是 否 在 10 一 50 及 25 一 28 
之 间 ， 查 询 结果 如 图 10.11 所 示 。 (实例 位 置 : 光盘 \TM\Instance\10\10.6) 

从 查询 结果 中 可 以 看 出 ， 在 范围 内 则 返回 1， 否 则 返回 0， 空 值 返 回 NULL。 

6. 运算 符 “IN” 

“IN” 用 于 判断 数据 是 否 存在 于 某 个 集合 中 。 其 表达 式 如 下 : 

x1 IN( 值 1, 值 2,…, 值 n) 

如 果 x1l 等 于 值 1 到 值 n 中 的 任何 一 个 值 ， 结 果 将 返回 1。 如 果 不 是 ， 结 果 将 返回 0。 

例 10.7 运用 “IN” 运 算 符 判 断 也 _book 表 中 row 字段 的 值 是 否 在 指定 的 范围 内 ， 查 询 结果 如 图 10.12 
所 示 。《〈 实 例 位 置 : 光盘 \TM\Instance\10\10.7) 

从 查询 结果 可 知 ， 在 范围 内 则 返回 1， 和 否则 返回 0， 空 值 返回 NULL。 
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二 箔 入 


运 身 符 


图 10.11 使 用 “BETWEEN AND” 运 算 符 判断 row 字段 值 的 范围 ”图 10.12 ”使 用 “IN” 运算 符 判断 row 字段 值 的 范 
7. 运算 符 “LIKE” 
“LIKE” 用 来 匹配 字符 串 。 其 表达 式 如 下 : 
x1 LIKE s1 


如 果 xl 与 字符 串 sl 匹配， 结果 将 返回 1。 否 则 返 
例 10.8 使 用 
图 10.13 所 示 。 


可 0。 

“LIKE” 运 算 符 判断 tb_book 表 中 的 user 字段 值 
(实例 位 置 ， 光盘 \TMNVInstance\10\10.8) 
从 查询 结果 可 知 ，user 字段 值 为 mr 字符 的 记录 ， 结 果 则 返回 1 
的 记录 ， 匹 配 则 返回 1， 否 则 返回 0。 
8. 运算 符 “REGEXP” 

“REGEXP” 
x1 REGEXP ' 匹 配方 式 ' 


司 样 用 于 匹配 字符 串 ， 但 其 使 用 的 是 正则 表达 式 进行 匹配 。 其 表达 式 格式 如 下 : 
如 果 xl 满足 匹配 方式 ， 结 果 


旦 . 下 
是 否 


与 指定 的 字符 串 


匹配 ， 查 询 结果 如 


否则 返回 0; user 字段 值 中 包含 1 字符 


将 返回 1; 
例 10.9 使 用 


否则 将 返回 0。 
“REGEXP” 


是 否 包含 指 


云 算 符 来 匹配 user 字段 的 值 是 否 以 指定 字符 开头 、 结 尾 ， 同 时 
定 的 字符 串 ， 执 行 结果 如 图 10.14 所 示 。【〔 实 例 位 置 ， 光盘 \TM\Instance\10\10.9) 


图 10.13 使 用 “LIKE” 


”运算 符 判 断 user 字段 是 否 匹 配 某 字符 
本 例 使 用 “REGEXP” 运 算 
尾 ; 在 user 字段 值 中 


图 10.14 使 用 “REGEXP” 运 算 符 匹配 字符 串 

运算 符 判 断 tb_book 表 中 user 字段 的 值 ， 是 否 以 mm 字符 开头 ; 是否 以 g 字符 结 
是 否 包 含 m 字符 ， 如 果 满 足 条 件 则 返回 1， 否 则 返回 0。 

[3 使 用 


“REGEXP” 运算 符 匹 配 字符 串 ， 其 使 用 方法 非常 简单 。“REGEXP” 运算 符 经 常 与 “^ 
“$” 和 “.” 一 起 使 用 。“^” 用 来 匹配 字符 串 的 开始 部 分 ; “$” 用 来 匹配 字符 串 的 结尾 部 分 
来 代表 字符 串 中 的 一 个 字符 。 


10.3.3 ”还 辑 运算 符 


锋 4 视频 讲解 :光盘 \TM\Video\ 第 10 章 \ 均 辑 运算 符 .exe 

逻辑 运算 符 用 来 判断 表达 式 的 真 假 。 如 果 表 达 式 是 真 ， 结 果 返 回 1。 如 果 表 达 式 是 假 ， 结 果 返 回 0。 逻 
辑 运算 符 又 称 为 布尔 运算 符 。MySQL 中 支持 4 种 逻辑 运算 符 ， 分 别 是 与 、 或 、 非 和 异 或 。 下 面 是 4 种 逻辑 
运算 符 的 符号 及 作用 ， 如 表 10.9 所 示 。 
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表 10.9 有 逻辑 运算 符 


符 ”号 作 用 
&& 或 AND 与 
| 或 OR 或 

! 或 NOT 非 
XOR, 异 或 


1. 与 运算 

“&&” 或 者 “AND” 是 与 运算 的 两 种 表达 方式 。 如 果 所 有 数据 不 为 0 且 不 为 空 值 (NULL) 时 ， 结 果 
返回 1; 如 果 存 在 任何 一 个 数据 为 0 时， 结果 返回 0;， 如 果 存在 一 个 数据 为 NULL 且 没 有 数据 为 0 时， 结果 
返回 NULL。 与 运算 符 支持 多 个 数据 同时 进行 运算 。 

例 10.10 运用 “&&” 运 算 符 判 断 row 字段 的 值 是 否 存 在 0 或 者 NULL (“row&&1” (row 字段 值 与 
1) 和 “row&&0” (row 字段 值 与 0) ) ， 如 果 存 在 则 返回 1， 否 则 返回 0， 空 值 返回 NULL。 执 行 结果 如 
图 10.15 所 示 。《〔 实 例 位 置 ， 光盘 \TM\Instance\10\10.10) 

结果 显示 ，“row&&1” 中 ，row 字段 的 值 为 非 0， 则 返回 1; row 字段 的 值 为 NULL， 则 返回 NULL; 
“roW&&0” 中 包含 0， 所 以 返回 值 为 0。 

2. 或 运算 

“I” 或 者 “OR” 表示 或 运算 。 所 有 数据 中 存在 任何 一 个 数据 不 为 非 0 的 数字 时 ， 结 果 返 回 1; 如 果 数 
据 中 不 包含 非 0 的 数字 ,但 包含 NULL 时 ， 结 果 返 回 NULL; 如 果 操 作 数 中 只 有 0 时， 结果 返回 0。 或 运算 
符 中 ”也 可 以 同时 操作 多 个 数据 

例 10.11 运用 “OR” 运 判断 tb_book 表 中 row 是 否 包含 NULL 或 者 非 0 数字 (“row OR 1” 和 
“row OR 0”) 。 执 行 结果 如 图 10.16 所 示 。〔 实 例 位 置 ， 光盘 \TM\Instance\10\10.11) 


图 10.15 使 用 “&&” 运 算 符 判断 数据 图 10.16 使 用 “OR” 运算 符 匹 配 数据 

结果 显示 ，“row OR 1” 中 包含 NULL 和 1 这 个 非 0 的 数字 ， 所 以 返回 结果 为 1; “row OR 0” 中 包含 
非 0 的 数字 、NULL 和 0 的 数字 ， 所 以 返回 NULL 和 1。 

3. 非 运算 

“1!” 或 者 “NOT” 表 示 非 运算 。 通 过 非 运算 ， 将 返回 与 操作 数据 相反 的 结果 。 如 果 操 作 数 据 是 非 0 的 
数字 ， 结 果 返 回 0; 如 果 操作 数据 是 0， 结 果 返 回 1; 如 果 操 作 数 据 是 NULL， 结 果 返 回 NULL。 

例 10.12 运用“!” 运 算 符 判断 也 _book 表 中 row 字段 的 值 是 否 为 0 或 者 NULL。 执 行 结果 如 图 10.17 
所 示 。 《实例 位 置 ; 光盘 \TMNVInstance\10\10.12) 

结果 显示 ，row 字段 中 值 为 NULL 的 记录 ， 返 回 值 为 NULL; 不 为 0 的 记录 ， 返 回 值 为 0。 

4. 异 或 运算 

“XOR” 表 示 异 或 运算 。 只 要 其 中 任何 一 个 操作 数据 为 NULL 时 ， 结 果 返 回 NULL; 如 果 xl 和 x2 中 
一 个 是 非 0， 另 一 个 是 0 时， 结果 返回 1。 

例 10.13 使 用 “XOR” 运 算 符 判 断 tb_book 表 中 字段 row 的 值 是 否 为 NULL (“row XOR 1” 和 “row 
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XOR 0”) 。 执 行 结果 如 图 10.18 所 示 。 (实例 位 置 : 光盘 \TM\Instance\10\10.13) 


DR 1 


1 


图 10.17 使 用 “!” 运 算 符 判断 数据 图 10.18 使 用 “XOR” 运 算 符 判断 数据 


结果 显示 , “row XOR 1” 中 row 字段 中 的 值 为 非 0 数字 和 NULL 值 ， 所 以 返回 值 为 0 和 NULL; “row 
XOR 0” 中 包含 0， 所 以 返回 值 为 1， 而 row 字段 值 为 NULL 的 记录 ， 返 回 值 则 为 NULL。 


10.3.4 ”位 运算 符 


位 运算 符 是 在 二 进 制 数 上 进行 计算 的 运算 符 。 位 运算 会 先 将 操作 数 变 成 二 进 制 数 ， 进 行 位 运算 。 然 后 
再 将 计算 结果 从 二 进 制 数 变 回 十 进 制 数 。MySQL 中 支持 6 种 位 运算 符 , 分 别 是 按 位 与 、 按 位 或 、 按 位 取 反 、 
按 位 异 或 、 按 位 左 移 和 按 位 右 移 。6 种 位 运算 符 的 符号 及 作用 如 表 10.10 所 示 。 

表 10.10 ”位 运算 符 

符 ” 号 作 用 

按 位 与 。 进 行 该 运算 时 ， 数 据 库 系统 会 先 将 十 进 制 的 数 转换 为 二 进 制 的 数 。 然 后 对 应 操作 数 的 每 个 二 进 
制 位 上 进行 与 运算 。1 和 1 相 与 得 1， 与 0 相 与 得 0。 运 算 完成 后 再 将 二 进 制 数 变 回 十 进 制 数 
按 位 或 。 将 操作 数 化 为 二 进 制 数 后 ， 每 位 都 进行 或 运算 。1 和 任何 数 进行 或 运算 的 结果 都 是 1，0 与 0 或 
运算 结果 为 0 
~ 按 位 取 反 。 将 操作 数 化 为 二 进 制 数 后 ， 每 位 都 进行 取 反 运算 。1 取 反 后 变 成 0，0 取 反 后 变 成 1 
按 位 异 或 。 将 操作 数 化 为 二 进 制 数 后 ， 每 位 都 进行 异 或 运算 。 相 同 的 数 异 或 后 结果 是 0， 不 同 的 数 异 或 


后 结果 为 1 

按 位 左 移 。“m<<n” 表 示 m 的 二 进 制 数 向 左 移 n 位 ， 右 边 补 上 mn 个 0。 例如， 二 进 制 数 001 左 移 1 位 后 
将 变 成 0010 

更 按 位 右 移 。“m>>n” 表 示 m 的 二 进 制 数 向 右 移 mn 位 ， 左 边 补 上 mn 个 0。 例 如， 二 进 制 数 011 右 移 1 位 后 


变 成 001， 最 后 一 个 1 直接 被 移出 
10.3.5 “运算 符 的 优先 级 


由 于 在 实际 应 用 中 可 能 需要 同时 使 用 多 个 运算 符 。 这 就 必须 考虑 运算 符 的 运算 顺序 。 正 所 谓 : “ 闻 道 
有 先后 ， 术 业 有 专攻 。” 
本 小 节 将 具体 曾 述 MySQL 运算 符 使 用 的 优先 级 ， 如 表 10.11 所 示 。 按 照 从 高 到 低 、 从 左 到 右 的 级 别 进 
行 运算 操作 。 如 果 优 先 级 相同 ， 则 表达 式 左边 的 运算 符 先 运算 。 
表 10.11 MySQL 运算 符 的 优先 级 
运 算 符 
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续 表 

优 先 级 运 算 符 

3 

4 */ DIV.%.MOD 

5 十 - 

6 >>.<< 

& 

8 | 

9 =<=>,<,<=,>,>=,!=,O.,IN.,IS,NULL.LIKE.REGEXP 

10 BETWEEN AND.CASE.WHEN.THEN.ELSE 

11 NOT 

12 &&.AND 

13 |.OR.XOR 

14 汪 


10.4 实 战 
区 4 视频 讲解 : 光盘 \TM\Video\ 第 10 章 \ 实 战 .exe 


10.4.1 查看 存储 引擎 和 创建 数据 库 


例 10.14 登录 数据 库 系 统 后 ， 创 建 student 数据 库 和 teacher 数据 库 。 都 创建 成 功 后 ， 然 后 查看 数据 库 
系统 中 还 存在 哪些 数据 库 ， 并 且 查 看 该 数据 库 系 统 支持 的 存储 引擎 类 型 。《〈 实 例 位 置 ， 光盘 \TMNInstancev 
10\10.14) 


代码 如 下 : 

Create database student; /创建 数据 库 
Create database teacher; 

Show databases; // 查 看 数据 库 
Show engines; /查看 存储 引擎 


运行 结果 如 图 10.19 所 示 。 
10.4.2 ”位 运算 的 比较 


例 10.15 本 实例 是 将 数字 4 和 6 进行 按 位 与 、 按 位 或 , 并 
将 4 按 位 取 反 。 位 运算 符 是 在 二 进 制 数 上 进行 计算 的 运算 符 。 
位 运算 会 先 将 操作 数 变 成 二 进 制 数 ， 进 行 位 运算 。 然 后 再 将 计 
算 结果 从 二 进 制 数 变 回 十 进 制 数 。《〈 实 例 位 置 : 光盘 \TM 
Instance\10\10.15) 

代码 如 下 : 

mysql>select 4&6,4|6,~4; 

运行 结果 如 图 10.20 所 示 。 


10.19 查询 存储 引擎 和 创建 数据 库 
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10.4.3 ”逻辑 运算 的 使 用 


例 10.16 将 数字 2、0 和 null 之 间 的 任意 两 个 进行 逻辑 运算 ， 逻 辑 运 算 符 用 来 判断 表达 式 的 真 假 。 如 
果 表达 式 是 真 ， 结 果 返 回 1; 如 果 表 达 式 是 假 ， 结 果 返 回 0。 风 辑 运 算 符 又 称 为 布尔 运算 符 。〔 实 例 位 置 : 
光盘 \TM\Instance\10\10.16) 

关键 代码 参考 如 下 : 

mysql>select 2&&0,2&&null,0 and null,2||0,2|Inull,0 or null; 

运算 结果 如 图 10.21 所 示 。 


10.4.4 浮 点 数 类 型 


例 10.17 某 表 的 字段 ab 和 的 数据 类 型 分 别 是 FLOAT、DOUBLE 和 DECIMAL, 向 表 中 插入 3.143、 
3.145、3.1434， 然 后 查询 该 表 的 数据 。《〈 实 例 位 置 : 光盘 \TM\Instance\10\10.17) 

代码 如 下 : 

Select * from aa; 

运行 结果 如 图 10.22 所 示 。 


图 10.20 位 运算 的 比较 图 1021 逻辑 运算 的 使 用 图 1022 数据 类 型 的 显示 
10.5 小 结 


本 章 对 MySQL 存储 引擎 、 数 据 类 型 、 运 算 符 进行 了 详细 讲解 ， 并 通过 举例 说 明 ， 使 读者 更 好 地 理解 所 
学 知识 的 用 法 。 在 阅读 本 章 时 ， 读 者 应 该 重点 掌握 什么 类 型 的 表 适 合 什么 类 型 的 存储 引擎 ， 数 据 类 型 及 各 
种 运算 符 的 使 用 ， 同 时 对 MySQL 中 的 数据 类 型 也 要 有 一 定 的 了 解 。 读 者 需要 认真 学 习 这 部 分 的 内 容 ， 位 运 
算 符 是 本 章 的 难点 。 因 为 ， 位 运算 符 需 要 将 操作 数 转换 为 二 进 制 数 ， 然 后 进行 位 运算 。 这 要 求 读者 能 够 掌 
握 二 进 制 运算 的 相关 知识 。 


10.6 ”学习 成 果 检 验 


1. 使 用 算术 运算 符 计算 5*6/6 的 值 。〔 实 例 位 置 : 光盘 \TMNInstance\10\10.18) 

2. 在 MySQL 中 执行 下 面 的 比较 运算 符 的 表达 式 : 40>=30，40<=30，NULL<=>NULL，7<=>7。 ( 实 
例 位 置 ， 光盘 \TMNInstance\10\10.19) 

3. 在 MySQL 中 执行 下 列 位 运算 : 11&15, 11|15, 13^15, 一 15。( 实 例 位置 : 光盘 \TMNInstance\10\10.20) 

4. MySQL 中 什么 数据 类 型 能 够 储存 路 径 ? 〈 实 例 位 置 : 光盘 \TMNVImnstance\10\10.21) 
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( 铝 / 视频 讲解 : 26 分 钟 ) 


My5QL 数据 库 中 提供 了 很 丰富 的 函数 。My5QL 函数 包括 数学 函 
数 、 字 符 串 函数 、 日 期 和 时 间 函 数 、 条 件 判 新 函 数 、 系 统 信 息 函 数 、 
加 密 函 数 、 格 式 化 函数 等 。 通 过 这 些 函 数 ， 可 以 简化 用 户 的 操作 。 例 
如 ， 字 符 串 连接 函数 可 以 很 方便 地 将 多 个 字符 串 连 接 在 一 起 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

mm 了 解 MySQL 国 数 

WM 了 解 MySQL 数学 困 数 的 使 用 方法 

WI 掌握 MySQL 字符 捉 困 数 的 使 用 方法 

H 掌握 MySQL 日 期 和 时 间 国 数 的 使 用 

WI 党 握 条 件 判断 国 数 的 应 用 

WI 党 握 系 统 国 数 和 加 密 印 数 的 使 用 

WI 掌握 其 他 图 数 的 使 用 
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11.1 MySQL 函数 


MySQL 函数 是 MySQL 数据 库 提 供 的 内 置 函 数 。 这 些 内 置 函数 可 以 帮助 用 户 更 加 方便 地 处 理 表 中 的 数 
据 。 本 节 中 将 简单 地 介绍 MySQL 中 包含 哪些 类 别 的 函数 ， 以 及 这 些 函数 的 使 用 范围 和 作用 。MySQL 中 内 
置 的 函数 及 其 作用 如 表 11.1 所 示 。 


表 11.1 MySQL 内 置 函 数 类 别 及 作用 


函数 作 用 

数学 函数 用 于 处 理 数字 。 这 类 函数 包括 绝对 值 函数 、 正 弦 函 数 、 余 弦 函 数 和 获取 随机 数 函 数 等 

字符 串 函数 用 于 处 理 字符 串 。 其 中 包括 字符 串 连接 函数 、 字 符 串 比较 函数 、 字 符 串 中 字母 大 小 写 转换 函数 等 

日 期 和 时 间 函 数 用 于 处 理 日 期 和 时 间 。 其 中 包括 获取 当前 时 间 的 函数 、 获 取 当 前 日 期 的 函数 、 返 回 年 份 的 函数 和 
返回 日 期 的 函数 等 

条 件 判断 函数 用 于 在 SQL 语句 中 控制 条 件 选择 。 其 中 包括 正 语句 、CASE 语句 和 WHEN 语句 等 

系统 信息 函数 用 于 获取 MySQL 数据 库 的 系统 信息 。 其 中 包括 获取 数据 库 名 的 函数 、 获 取 当 前 用 户 的 函数 和 获 
取 数 据 库 版 本 的 函数 等 

加 密 函 数 用 于 对 字符 串 进行 加 密 解 密 。 其 中 包括 字符 串 加 密 函 数 和 字符 串 解密 函数 等 

其 他 函数 包括 格式 化 函数 和 锁 函 数 等 


MySQL 的 内 置 函数 不 但 可 以 在 SELECT 查询 语句 中 应 用 ,同样 也 可 以 在 INSERT、UPDATE 和 DELETE 


等 语句 中 应 用 。 例 


如 ， 在 INSERT 添加 语句 中 ， 应 用 日 期 时 间 函 数 获取 系统 的 当前 时 间 ， 并 且 将 其 添加 到 


数据 表 中 。MySQL 内 置 函 数 可 以 对 表 中 数据 进行 相应 地 处 理 ， 以 便 得 到 用 户 希望 得 到 的 数据 。 有 了 这 些 内 
置 函数 可 以 使 MySQL 数据 库 的 功能 更 加 强大 。 下 面 将 对 MySQL 的 内 置 函数 逐一 进行 详细 介绍 。 


11.2 数学 函数 


敬 4 视频 讲解 : 光盘 \TMIVVideo\ 第 11 章 \ 数 学 函数 .exe 
数学 函数 是 MySQL 中 常用 的 一 类 函数 。 其 主要 用 于 处 理 数 字 ， 包 括 整 型 和 浮 点 数 等 。MySQL 中 内 置 


的 数学 函数 及 其 作用 如 表 11.2 所 示 。 
表 11.2 MySQL 的 数学 函数 
函数 作 用 
ABS(x) 返回 x 的 绝对 值 
CEIL(x).CEILIN(x) 返回 不 小 于 x 的 最 小 整数 值 
FLOOR(x) 返回 不 大 于 x 的 最 大 整数 值 
RANDO 返回 0~1 的 随机 数 
RAND(x) 返回 0~1 的 随机 数 ，x 值 相 同时 返回 的 随机 数 相同 
SIGN(x) 返回 参数 作为 -1、0 或 1 的 符号 ， 该 符号 取决 于 x 的 值 为 负 、 零 或 正 
PIO 返回 圆周 率 的 值 。 默 认 的 显示 小 数位 数 是 7 位 ， 然 而 MySQL 内 部 会 使 用 完全 双 精 度 值 
TRUNCATE(x.y) 返回 数值 x 保留 到 小 数 点 后 y 位 的 值 
ROUND(x) 返回 离 x 最 近 的 整数 
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续 表 
函数 作 用 
ROUND(x.y) 保留 x 小数点 后 y 位 的 值 ， 但 截断 时 要 进行 四 合 五 入 
POW(x.y).POWER(x.y) | 返回 x 的 y 乘 方 的 结果 值 
SQRT() 返回 非 负 数 x 的 二 次 方 根 
EXP(x) 返回 e 的 x 乘 方 后 的 值 ( 自 然 对 数 的 底 ) 
MOD(x.y) 返回 x 除 以 y 后 的 余数 
LOG(x) 返回 x 的 基数 为 2 的 对 数 
LOG10(x) 返回 x 的 基数 为 10 的 对 数 
RADIANS(x) 将 角度 转换 为 弧度 
DEGREES(x) 返回 参数 x， 该 参数 由 弧度 被 转化 为 度 
SIN(x) 返回 x 正弦， 其 中 x 在 弧度 中 被 给 定 
ASIN(x) 返回 x 的 反正 弦 ， 即 正弦 为 x 的 值 。 若 x 不 在 -1 到 1 的 范围 之 内 ， 则 返回 NULL 
COS(x) 返回 x 的 余弦 ， 其 中 x 在 弧度 上 已 知 
ACOS(x) 返回 x 反 余弦 ， 即 余弦 是 x 的 值 。 若 x 不 在 -1 到 1 的 范围 之 内 ， 则 返回 NULL 
TAN(x) 返回 x 的 反正 切 ， 即 正切 为 x 的 值 
返回 两 个 变量 x 及 y 的 反正 切 。 它 类 似 于 y 或 x 的 反正 切 计算 ， 除 非 两 个 参数 的 符号 均 用 
ATAN(%),ATAN2(xy) | 于 确定 结果 所 在 象限 
COT(x) 返回 x 的 余 切 


下 面 对 其 中 的 常用 函数 进行 讲解 ， 并 且 配 合 以 实例 做 详细 说 明 。 
11.2.1 ABS(x) 函 数 


ABS(x) 函 数 用 于 求 绝 对 值 。 

例 11.1 使 用 ABS(x) 函 数 来 求 8 和 -8 的 绝对 值 。 (实例 位 置 : 光盘 \TM\Instance\11\11.1) 
其 语句 如 下 : 

select ABS(8),ABS(-8); 

查询 结果 如 图 11.1 所 示 。 


1 


-eh 


.2.2 “FLOOR(x) 函 数 


FLOOR(x) 函 数 返 回 小 于 或 等 于 x 的 最 大 整数 。 

例 11.2 应 用 FLOOR(x) 函 数 求 出 3.6 和 -4 的 最 大 整数 。〈 实 例 位 置 ， 光盘 \TM\Instance\11\11.2) 
其 语句 如 下 : 

select FLOOR(3.6),FLOOR(-4); 

其 查询 结果 如 图 11.2 所 示 。 


BS C8 ABS 


图 11.1 使 用 ABS(x) 求 绝对 值 图 11.2 使 用 FLOOR(x) 函 数 求 出 最 大 整数 
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11.2.3 ”RAND() 函 数 


RAND0 函 数 是 返回 0 一 1 的 随机 数 。 但 是 RANDO 返 回 的 数 是 完全 随机 的 。 

例 11.3 运用 RAND0 函 数 ， 获 取 两 个 随机 数 。 (实例 位 置 : 光盘 \TM\Instance\11\11.3) 
其 语句 如 下 : 

Select RAND(),RAND(); 

其 查询 结果 如 图 11.3 所 示 。 


11.2.4 ”PI() 函 数 


PIO 函 数 用 于 返回 圆周 率 。 

例 11.4 使 用 PIO 函 数 获取 圆周 率 。〈 实 例 位 置 ， 光盘 \TM\Instance\11\11.4) 
其 语句 如 下 : 

select PI(); 

查询 结果 如 图 11.4 所 示 。 


11.2.5 TRUNCATE(x,y) 函 数 


TRUNCATE(x,y) 函 数 返回 x 保留 到 小 数 点 后 y 位 的 值 。 

例 11.5 使 用 TRUNCATE(x,y) 函 数 返回 3.987654 小 数 点 后 3 位 的 值 。( 实 例 位 置 : 光盘 \TM\Instance\ 
11\11.5) 

其 语句 如 下 : 

select TRUNCATE(3.987654,3); 

其 查询 结果 如 图 11.5 所 示 。 


图 11.3 使 用 RANDO 函 数 获取 随机 数 。 图 11.4 使 用 PIO 函数 获取 圆周 率 ”图 11.5 使 用 TRUNCATE(x.y) 函 数 获取 数据 


11.2.6 ROUND(x) 函 数 和 ROUND(x,y) 函 数 


ROUNDG) 函 数 返 回 离 x 最 近 的 整数 ， 也 就 是 对 x 进行 四 舍 五 入 处 理 ;， ROUND(x.y) 函 数 返 回 x 保留 到 
小 数 点 后 y 位 的 值 ， 截 断 时 需要 进行 四 舍 五 入 处 理 。 

例 11.6 使 用 ROUND(x) 函 数 获取 2.7 和 1.8 最近 的 整数 ,使 用 ROUND(x,y) 函 数 获 取 3.123456 小 数 点 
后 3 位 的 值 。〈 实 例 位 置 : 光盘 \TM\Instance\11\11.6) 

其 语句 如 下 : 

select ROUND(2.7),ROUND(1.8),ROUND(3.123456,3); 

查询 结果 如 图 11.6 所 示 。 
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11.2.7 ”SQRT(x) 函 数 


SQRT(x) 函 数 用 于 求 平方 根 。 


例 11.7 使 
其 语句 如 下 : 


SQRT(x) 函 数 求 36 和 25 的 平方 根 。 


(实例 位 置 :光盘 \TM\Instance\11\11.7) 


select SQRT(36),SQRT(25); 


其 查询 结果 如 图 11.7 所 示 。 


ROUNDCL 


ov in set (9.08 se 


图 11.6 使 用 ROUND(x) 函 数 和 ROUND(x.y) 函 数 获取 数据 


图 11.7 使 用 SQRT(x) 函 数 求 16 和 25 的 平方 根 


11.3 字符 串 函数 


句 4 视频 讲解 : 光盘 \TM\Video\ 第 11 章 \ 字 符 串 函数 .exe 
字符 串 函 数 是 MySQL 中 最 常用 的 一 类 函数 。 字 符 串 函数 主要 用 于 处 理 表 中 的 字符 串 。MySQL 内 置 的 
字符 串 函 数 及 其 作用 如 表 11.3 所 示 。 


函数 


表 11.3 MySQL 的 字符 串 函 数 
作 用 


返回 字符 串 s 的 字符 数 


CHAR_LENGTHG) 


LENGTHCS) 


返回 值 为 字符 串 s 的 长 度 ， 单 位 为 字 节 。 一 个 多 字 节 字符 算 作 多 字 节 。 这 意味 着 对 于 一 
个 包含 5 个 两 字 节 字符 的 字符 串 ，LENGTHO 的 返回 值 为 10， 而 CHAR_LENGTHO 的 
返回 值 则 为 5 


CONCAT(s1.s2....) 


返回 结果 为 连接 参数 产生 的 字符 串 。 如 有 任何 一 个 参数 为 NULL， 则 返回 值 为 NULL。 
或 许 有 一 个 或 多 个 参数 。 如 果 所 有 参数 均 为 非 二 进 制 字符 串 ， 则 结果 为 非 二 进 制 字符 
串 。 如 果 自 变量 中 含有 任 一 二 进 制 字符 串 ， 则 结果 为 一 个 二 进 制 字符 串 。 一 个 数字 参 
数 被 转化 为 与 之 相等 的 二 进 制 字符 串 格 式 ， 若 要 避免 这 种 情况 ， 可 使 用 显 式 类 型 cast， 
如 SELECT CONCAT(CAST(int col AS CHAR). char col) 


CONCAT WS(x.s1.s2....) 


同 CONCAT(s1.s2,.….) 函 数 ， 但 是 每 个 字符 串 直 接 要 加 上 x 


INSERT(s1.x.len.s2 将 字符 串 s2 替换 s1 的 x 位 置 开始 长 度 为 len 的 字符 串 

UPPER(s).UCASE(s) 将 字符 串 s 的 所 有 字母 都 变 成 大 写字 母 

LOWER(s).LCASE(s) 将 字符 串 s 的 所 有 字母 都 变 成 小 写字 母 

LEFT(s.n) 返回 从 字符 串 s 开始 的 最 左边 n 个 字符 

RIGHT(s.n) 返回 从 字符 串 s 开始 的 最 右边 n 个 字符 

i 返回 字符 串 sl, 其 左边 由 字符 串 s2 填补 到 len 字符 长 度 。 假 如 sl 的 长 度 大 于 len， 则 
返回 值 被 缩短 至 len 字符 
返回 字符 串 s1, 其 右边 被 字符 串 s2 填补 至 len 字符 长 度假 如 字符 串 sl 的 长 度 大 于 len， 


RPAD(s1.len.s2) 
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则 返回 值 被 缩短 到 与 len 字符 相同 长 度 
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续 表 
函数 作 用 
LTRIMGS) 返回 字符 串 s， 其 引导 空格 字符 被 删除 
RIRIM(S) 返回 字符 串 s， 结 尾 空格 字符 被 删 去 
TRIMG) 去 掉 字符 串 s 开始 处 和 结尾 处 的 空格 
TRIM(s1 FROM S) 去 掉 字符 串 s 中 开始 处 和 结尾 处 的 字符 串 sl 
REPEAT(s.n) 将 字符 串 s 重复 n 次 
SPACE(n) 返回 n 个 空格 
REPLACE(s.s1.s2) 用 字符 串 s2 葵 代 字符 串 s 中 的 字符 串 s1 
STRCMP(s1.s2) 比较 字符 串 sl1 和 s2 
SUBSTRING(s.n.len) 获取 从 字符 串 s 第 n 个 位 置 开始 长 度 为 len 的 字符 串 
MID(s.n.len) 同 SUBSTRING(s,n,len) 
LOCATEG1.s)POSITION | /字符 串 s 中 获取 sl 的 开始 位 置 
GLINS) 
INSTR(s.s1) 从 字符 串 s 中 获取 sl 的 开始 位 置 
REVERSE(s) 将 字符 串 s 的 顺序 反 过 来 
ELT(n.s1,s2,...) 返回 第 n 个 字符 串 
返回 一 个 字符 串 ， 生 成 规则 如 下 : 针对 bits 的 二 进 制 格式 ， 如 果 其 位 为 1， 则 返回 一 个 
on 值 ， 如 果 其 位 为 0， 则 返回 一 个 o 全 值 。 每 个 字符 串 使 用 separator 进行 分 隔 ， 默 认 值 
EXPORT_SET(bits.on.off[， | 为 “,”。number_of bits 参数 指定 bits 可 用 的 位 数 ， 默 认 值 为 64 位 。 例 如 ， 生 成 数字 
separator[number of bits]]) | 182 的 二 进 制 (10110110) 蔡 换 格式 ， 以 “@” 作 为 分 隔 符 ， 设 置 有 效 位 为 6 位 。 其 语 
名 如 下 : select EXPORT SET(182.YN.Q@'.6): 
其 运行 结果 为 : N@Y@Y@N@Y@Y 
FIELD(s.s1.s2....) 返回 第 一 个 与 字符 串 s 匹配 的 字符 串 的 位 置 
FIND IN SET(s1.s2) 返回 在 字符 串 s2 中 与 s1 匹配 的 字符 串 的 位 置 
MAKE SET(x.s1.s2....) 按 x 的 二 进 制 数 从 s1,s2,…,sn 中 选取 字符 串 


下 面 对 其 中 的 常用 函数 进行 讲解 ， 并 且 配 合 以 例子 做 详细 说 明 。 
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INSERT(s1,x,len,s2) 函 数 


INSERT(s1,x,len,s2) 函 数 将 字符 串 sl 中 x 位置 开始 长 度 为 len 的 字符 串 用 字符 串 s2 替换 。 
例 11.8 使 用 INSERTO 函 数 将 mrkj 字符 串 中 的 区 替换 为 book.( 实 例 位 置 :光盘 \TM\Instance\11\11.8) 


其 语句 如 下 : 


select INSERT('mrkj',3,2,'book"); 


蔡 换 后 的 查询 结果 如 图 11.8 所 示 。 


11.3.2 UPPER(s)、UCASE(s) 函 数 


UPPER(s) 函 数 和 UCASE(s) 函 数 将 字符 串 s 的 所 有 字母 变 


成 大 写字 母 。 
例 11.9 下 面 使 有 


图 11.8 使 用 INSERTO 函 数 蔡 换 指定 字符 串 


UPPER(s) 函 数 和 UCASE(s) 函 数 将 mingri 字符 串 中 的 所 有 字母 变 成 大 写字 母 。 (实例 


位 置 ， 光盘 \TM\Instance\11\11.9) 


_ 国 
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其 语句 如 下 : 
select UPPER('mingri"),UCASE('mingri"); 
其 转换 后 的 结果 如 图 11.9 所 示 。 


11.3.3 LEFT(s,n) 函 数 


LEFT(s,n) 函 数 返回 字符 串 s 的 前 n 个 字符 。 

例 11.10 ”应 用 LEFTO 函 数 返回 mingribook 字符 串 的 前 6 个 字符 (实例 位 置 :光盘 \TM\Instance\11\11.10) 
其 语句 如 下 : 

select LEFT('mingribook',6); 

其 截取 结果 如 图 11.10 所 示 。 


图 11.9 使 用 UPPER(s) 函 数 和 UCASE(s) 函 数 图 11.10 使 用 LEFT 函数 返回 指定 字符 
将 mingri 字符 串 中 的 所 有 字母 变 成 大 写字 母 
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mh 


.3.4 ”RTRIM(s) 函 数 


RTRIM(s) 函 数 将 去 掉 字 符 串 s 结尾 处 的 空格 。 

例 11.11 应 用 RTRIMO 函 数 去掉 mingri 结尾 处 的 空格 。 (实例 位 置 : 光盘 \TMNInstance\1l\11.11) 
其 语句 如 下 : 

select CONCAT(+',RTRIM(C mingri "),+"); 

其 结果 如 图 11.11 所 示 。 


11.3.5 ”SUBSTRING(s,n,len) 函 数 
SUBSTRING(s,n,len) 函 数 从 字符 串 s 的 第 n 个 位 置 开始 获取 长 度 为 len 的 字符 串 。 


例 11.12 下 面 使 用 SUBSTRING0O 函 数 获取 从 mingribook 字符 串 的 第 1 位 开始 的 6 个 字符 ， 结 果 如 
图 11.12 所 示 。【〔 实 例 位 置 ， 光盘 \TM\Instance\11\11.12) 


图 11.11 使 用 RTRIM 函数 去 掉 mingri 结尾 处 的 空格 图 11.12 使 用 SUBSTRING 函数 获取 指定 长 度 字符 串 
11.3.6 REVERSE(s) 函 数 


REVERSE(s) 函 数 将 字符 串 s 的 顺序 反 过 来 。 
例 11.13 ”下面 使 用 REVERSE0 函 数 将 mingri 字符 串 的 顺序 反 过 来 ， 结 果 如 图 11.13 所 示 。 (实例 位 
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置 : 光盘 \TM\Instance\11\11.13) 
11.3.7 FIELD(s,s1,s2,...) 函 数 
FIELD(s,s1,s2,…) 函 数 返 回 第 一 个 与 字符 串 s 匹配 的 字符 串 的 位 置 。 


例 11.14 应 用 FIELDO 函 数 返 回 第 一 个 与 字符 串 mr 匹配 的 字符 串 位 置 ,结果 如 图 11.14 所 示 。( 实 例 
位 置 ， 光盘 \TM\Instance\11\11.14) 


图 11.13 ”使 用 REVERSEO 函 数 将 mingri 图 11.14 使 用 FIELDO 函 数 返回 第 一 个 与 
字符 串 的 顺序 反 过 来 字符 串 mr 匹配 的 字符 串 位 置 


11.4 日 期 和 时 间 函 数 


区 J 视频 讲解 : 光盘 \TM\Video\ 第 11 章 \ 日 期 和 时 间 函 数 .exe 
日 期 和 时 间 函 数 是 MySQL 中 另 一 最 常用 的 函数 。 其 主要 用 于 对 表 中 的 日 期 和 时 间 数 据 的 处 理 . MySQL 
内 置 的 日 期 时 间 函 数 及 作用 如 表 11.4 所 示 。 


表 11.4 MySQL 的 日 期 和 时 间 函 数 


函数 作 用 
CURDATEOQ.CURRENT DATEOQ 返回 当前 日 期 
CURTIMEO.CURRENT TIMEOQ 返回 当前 时 间 
NOWO.CURRENT_TIMESTAMPO.LOCALT | 返回 当前 日 期 和 时 间 
IMEO.SYSDATEO.LOCALTIMESTAMP 
UNIX_TIMESTAMPO 以 UNIX 时 间 截 的 形式 返回 当前 时 间 
UNIX_TIMESTAMP(d) 将 时 间 d 以 UNIX 时 间 戳 的 形式 返回 
FROM UNIXTIME(d) 把 UNIX 时 间 截 的 时 间 转 换 为 普通 格式 的 时 间 
UTC DATEO 返回 UTC (Universal Coordinated Time， 国 际 协调 时 间 ) 日 期 
UTC TIMEO 返回 UTC 时 间 
MONTH(d) 返回 日 期 4 中 的 月 份 值 ， 范 围 是 1 一 12 
MONTHNAME(d) 返回 日 期 4 中 的 月 份 名 称 ， 如 January、February 等 
DAYNAME(d) 返回 日 期 4 是 星期 几 ， 如 Monday、Tuesday 等 
DAYOEFWEEK(d 返回 日 期 d 是 星期 几 ，1 表示 星期 日 ，2 表示 星期 一 等 
WEEKDAY(d) 返回 日 期 4 是 星期 几 ，0 表示 星期 一 ，1 表示 星期 二 等 
WEEK(d) 计算 日 期 4 是 本 年 的 第 几 个 星期 ， 范 围 是 0 一 53 
WEEKOFYEAR(d) 计算 日 期 4 是 本 年 的 第 几 个 星期 ， 范 围 是 1 一 53 
DAYOFYEAR(d) 计算 日 期 d 是 本 年 的 第 几 天 
DAYOFMONTH(d) 计算 日 期 d 是 本 月 的 第 几 天 
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续 表 

函数 作 用 
YEAR(d) 返回 日 期 4 中 的 年 份 值 
QUARTER(d) 返回 日 期 4 是 第 几 季度 ， 范 围 是 1 一 4 
HOUR() 返回 时 间 t 中 的 小 时 值 
MINUTE(D 返回 时 间 t 中 的 分 钟 值 
SECOND( 返回 时 间 t 中 的 秒 钟 值 
EXTRACT(type FROM d) 从 日 期 d 中 获取 指定 的 值 ，type 指定 返回 的 值 ， 如 YEAR、HOUR 等 
TIME TO SEC(t 将 时 间 t 转 换 为 秒 
SEC TO TIME(s) 将 以 秒 为 单位 的 时 间 s 转换 为 时 分 秒 的 格式 
TO DAYS(d) 计算 日 期 d~0000 年 1 月 1 日 的 天 数 
FROM DAYS(n) 计算 从 0000 年 1 月 1 日 开始 n 天 后 的 日 期 
DATEDIFF(d1.d2 计算 日 期 dl 一 d2 之 间 相 隔 的 天 数 
ADDDATE(d.n) 计算 起 始 日 期 d 加 上 n 天 的 日 期 
ADDDATE(d.INTERVAL expr type) 计算 起 始 日 期 4 加 上 一 个 时 间 段 后 的 日 期 
DATE ADD(dINTERVAL expr type) 同 ADDDATE(dINTERVAL ntype) 
SUBDATE(dn 计算 起 始 日 期 d 减 去 n 天 后 的 日 期 
SUBDATE(d,INTERVAL expr type) 计算 起 始 日 期 4 减 去 一 个 时 间 段 后 的 日 期 
ADDTIME(t.n) 计算 起 始 时 间 t 加 上 1n 秒 的 时 间 
SUBTIME(t.n) 计算 起 始 时 间 t 减 去 n 秒 的 时 间 
DATE FROMAT(d. 按照 表达 式 f 的 要 求 显示 日 期 d 
TIME FROMAT(tD 按照 表达 式 f 的 要 求 显示 时 间 t 
GET FORMAT(type.s) 根据 字符 串 s 获取 type 类 型 数据 的 显示 格式 


下 面 对 其 中 的 常用 函数 进行 讲解 ， 并 且 配 合 以 实例 做 详细 说 明 。 
11.4.1 CURDATE() 函 数 和 CURRENT_DATE() 函 数 


CURDATE0 和 CURRENT_DATE0 函 数 获取 当前 日 期 。 

例 11.15 ”下面 使 用 CURDATE0 和 CURRENT _DATE0O 函 数 获取 当前 日 期 。〈 实 例 位 置 : 光盘 \TM 
Instance\11\11.15) 

其 语句 如 下 : 

select CURDATE(),CURRENT_DATE(); 

其 查询 结果 如 图 11.15 所 示 。 


11.4.2 ”CURTIME() 函 数 和 CURRENT_TIME() 函 数 


CURTIME0 和 CURRENT_TIME0O 函 数 获取 当前 时 间 。 

例 11.16 下面 使 用 CURTIME0O 和 CURRENT _TIME0O 函 数 获取 当前 时 间 。《〔 实 例 位 置 ， 光盘 \TM 
Instance\11\11.16) 

其 语句 如 下 : 

select CURTIME(),CURRENT_TIME(); 

其 查询 结果 如 图 11.16 所 示 。 
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图 11.15 使 用 CURDATEO 和 CURRENT_ 图 11.16 使 用 CURTIMEO 和 CURRENT TIMEO 
DATEO 函 数 获取 当前 日 期 函数 获取 当前 时 间 


11.4.3 ”NOWO 函 数 


NOWO 函 数 获取 当前 日 期 和 时 间 。 还 有 CURRENT_TIMESTAMP0O、LOCALTIME0O、SYSDATE0 和 
LOCALTIMESTAMP0O 函 数 也 同样 可 以 获取 当前 日 期 和 时 间 。 

例 11.17 下 面 使 用 NOWO、CURRENT _TIMESTAMP0、LOCALTIMEO、SYSDATE0 函 数 来 获取 当前 
日 期 和 时 间 。《〈 实 例 位 置 ， 光盘 \TM\Instance\11\11.17) 


其 语句 如 下 : 
select NOW(),CURRENT_TIMESTAMP(),LOCALTIME(),SYSDATE(); 
运行 结果 如 图 11.17 所 示 。 


11.4.4 DATEDIFF(d1,d2) 函 数 
DATEDIFF(dl,d2) 函 数 用 于 计算 日 期 dl 与 d2 之 间 相隔 的 天 数 


例 11.18 使 用 DATEDIFF(d1,d2) 函 数 计算 2012-11-18 与 2018-11-11 之 间 相 隔 的 天 数 。【〈 实 例 位 置 ， 光 
盘 \TMNInstance\1l\11.18) 


其 语句 如 下 : 
select DATEDIFF('2012-11-18","2012-11-11"); 
结果 如 图 11.18 所 示 。 


图 11.17 使 用 NOWO、URRENT _TIMESTAMP0O 等 图 11.18 使 用 DATEDIFF(d1.d2) 函 数 计算 2012-11-18 
函数 获取 当前 日 期 和 时 间 与 2012-11-11 之 间 相 隔 的 天 数 


11.4.5 ADDDATE(d,n) 函 数 


ADDDATE(dn) 函 数 用 于 返回 起 始 日 期 4 加 上 nm 天 的 日 期 。 藤 
例 11.19 使 用 ADDDATE(dn) 函 数 返 回 2012-11-16 加 上 2 天 的 
日 期 ， 结果 如 图 11.19 所 示 。 (实例 位 置 : 光盘 TM\Instance\11\11.19》 ”图 111 使 用 ADDDATE(da 函 数 
返回 2012-11-16 加 上 2 天 的 日 期 
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11.4.6 ADDDATE(d,INTERVAL exprtype) 函 数 


ADDDATE(d,INTERVAL expr type) 函 数 返 回 起 始 日 期 4 加 上 一 个 时 间 段 后 的 日 期 。 

例 11.20 使 用 ADDDATE(dINTERVAL expr type) 函 数 返回 2012-11-16 加 上 1 年 2 个 月 后 的 日 期 。( 实 
例 位 置 ， 光盘 \TM\Instance\11\11.20) 

其 语句 如 下 : 

select ADDDATE(2012-11-16',INTERVAL '1 2' YEAR_MONTH); 

其 运行 结果 如 图 11.20 所 示 。 


11.4.7 SUBDATE(d,n) 函 数 


SUBDATE(dn) 函 数 返 回 起 始 日 期 d 减 去 n 天 的 日 期 。 
例 11.21 使 用 SUBDATE(dn) 函 数 返 回 2012-11-16 减 去 4 天 后 的 日 期 ， 结 果 如 图 11.21 所 示 。 (实例 
位 置 : 光盘 \TM\Instance\11\11.21) 


11.20 ”使 用 ADDDATE(d,INTERVAL expr type) 函 数 11.21 使 用 SUBDATE(d.n) 函 数 返 回 2012-11-16 
返回 2012-11-16 加 上 1 年 2 个 月 后 的 日 期 减 去 4 天 后 的 日 期 


11.5 条 件 判 断 函 数 


多 i 视频 讲解 : 光盘 \TM\Video\ 第 11 章 \ 条 件 判断 函数 .exe 
条 件 函 数 用 来 在 SQL 语句 中 进行 条 件 判断 。 根 据 不同 的 条 件 ， 执 行 不 同 的 SQL 语句 。MySQL 支持 的 
条 件 判 断 函数 及 作用 如 表 11.5 所 示 。 
表 11.5 MySQL 的 条 件 判断 函数 


函数 作 用 
IF(expr.v1.v2) 如 果 表 达 式 expr 成 立 ， 则 执行 V1; 否则 执行 v2 
IFNULL(v1.v2) 如 果 v1 不 为 空 ， 则 显示 v1 的 值 ， 否 则 显示 v2 的 值 


case 表示 函数 开始 ，end 表示 函数 结束 。 如 果 表 达 式 exprl 成 立 ， 则 返 
回 v1 的 值 ， 如 果 表 达 式 expr2 成 立 ， 则 返回 v2 的 值 。 依 次 类 推 ， 最 
后 遇 到 else 时 ， 返 回 vn 的 值 。 它 的 功能 与 PHP 中 的 switch 语句 类 似 
case 表示 函数 开始 ，end 表示 函数 结束 。 如 果 表 达 式 expr 取 值 为 el， 
则 返回 vl 的 值 ， 如 果 表 达 式 expr 取 值 为 e2， 则 返回 v2 的 值 ， 依 次 类 
推 ， 最 后 遇 到 else， 则 返回 vn 的 值 


CASE WHEN exprl THEN v1 [WHEN expr2 
THEN v2 ...][ELSE wn] END 


CASE expr WHEN el THEN v1 [WHEN e2 
THEN v2 ...J[ELSE wn] END 


例 11.22 查询 编程 词典 业绩 信息 表 ， 如 果 业 绩 超过 100 万 ， 则 输出 “Very Good”， 如 果 业 绩 小 于 100 
万 大 于 10 万 ， 则 输出 “Popularly”， 否 则 输出 “Not Good”。〔 实 例 位 置 ， 光盘 \TM\Instance\11\11.22) 
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其 语句 如 下 : 
select id,grade, CASE WHEN grade>1000000 THEN Very Good' WHEN grade<1000000 and grade >=100000 
THEN 'Popularly ELSE 'Not Good' END level from tb_bccd; 


其 查询 结果 如 图 11.22 所 示 。 


图 11.22 条 件 判断 函数 的 应 用 


11.6 系统 信息 函数 


系统 信息 函数 用 来 查询 MySQL 数据 库 的 系统 信息 。 例如， 查询 数据 库 的 版 本 、 查 询 数据 库 的 当前 用 户 
等 。 下 面 是 各 种 系统 信息 函数 的 作用 ， 如 表 11.6 所 示 。 
表 11.6 ”MySQL 的 系统 信息 函数 


函数 作 用 示例 
VERSIONO 获取 数据 库 的 版 本 号 select VERSIONO: 
CONNECTION_IDO 获取 服务 器 的 连接 数 select CONNECTION _IDO: 
DATABASE(Q.SCHEMAI 获取 当前 数据 库 名 select DATABASEOQ.SCHEMAO: 
USERO.SYSTEM_USERO.SESSION_USERO | 获取 当前 用 户 select USERO.SYSTEM_USERO: 
CURRENT USERO.CURRENT USER 获取 当前 用 户 select CURRENT USERO: 
CHARSET(str) 获取 字符 串 str 的 字符 集 select CHARSET(mrsoft): 
COLLATION(str 获取 字符 串 str 的 字符 排列 方式 select COLLATION(mrsoft): 
LAST INSERT IDO 获取 最 近 生 成 的 AUTO INCREMENT 值 |select LAST INSERT IDO: 


11.6.1 获取 MySQL 版 本 号 、 连 接 数 和 数据 库 名 的 函数 


VERSION0O 函 数 返 回 数据 库 的 版 本 号 ; CONNECTION ID0 函 数 返 回 服 
务 器 的 连接 数 , 也 就 是 到 现在 为 止 MySQL 服务 的 连接 次 数 ; DATABASEO 
和 SCHEMAO 返 回 当前 数据 库 名 。 

例 11.23 下面 将 演示 VERSIONO CONNECTION ID0.DATABASEO0 
和 SCHEMAO 4 个 函数 的 用 法 。〈 实 例 位 置 : 光盘 \TMNVInstance\1l\11.23) 

其 中 , VERSIONO 函 数 返 回 的 版 本 号 为 “5.5.12”; CONNECTOIN IDO 
返回 的 连接 数 为 30;DATABASE0 和 SCHEMA0 返 回 的 当前 数据 库 名 是 test。 


11.6.2 ”获取 用 户 名 的 函数 


图 11.23 获取 MySQL 版 本 号 、 
连接 数 和 数据 库 名 得 函数 


USERO、SYSTEM_USERO、SESSION_ USERO、CURRENT_USERO 和 CURRENT_USER 这 几 个 函数 
可 以 返回 当前 用 户 的 名 称 。 
例 11.24 ”下面 查询 当前 用 户 的 用 户 名 。 运 行 结果 如 图 11.24 所 示 .( 实 例 位 置 :光盘 \TM\Instance\11\11.24) 
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结果 显示 ， 当 前 用 户 的 用 户 名 为 root。Localhost 是 主机 名 。 因 为 服务 器 和 客户 端 在 一 台 机 器 上 ， 所 以 
服务 器 的 主机 名 为 localhost。 用 户 名 和 主机 名 之 间 用 符号 “@” 进 行 连接 。 
11.6.3 ”获取 字符 串 的 字符 集 和 排序 方式 的 函数 

CHARSET(str) 函 数 返 回 字符 串 str 的 字符 集 ， 一 般 情况 下 这 个 字符 集 就 是 系统 的 默认 字符 集 ; 


COLLATION( 人 st 函数 返回 


例 11.25 


字符 串 str 的 字符 排列 方式 。 


下 面 查看 字符 串 “aa” 的 字符 集 和 字符 串 排序 方式 。 运 行 结 果 如 图 11.25 所 示 。 


光盘 \TM\Instance\11\11.25) 


elocalhost 


图 11.24 获取 用 户 名 的 函数 


《实例 位 置 : 


图 11.25 ”获取 字符 串 的 字符 集 和 排序 方式 的 函数 


117 加 密 通 数 
加 密 函 数 是 MySQL 中 用 来 对 数据 进行 加 密 的 函数 。 因为 数据 库 中 有 些 很 敏感 的 信息 不 希望 被 其 他 人 看 
到 ， 所 以 就 可 以 通过 加 密 的 方式 来 使 这 些 数据 变 成 看 似 乱码 的 数据 。 例 如 ， 用 户 密码 就 应 该 进行 加 密 。 各 
种 加 密 函数 的 作用 如 表 11.7 所 示 。 
表 11.7 MySQL 的 加 密 函 数 
函数 作 用 示 例 
对 字符 串 str 进行 加 密 。 经 此 函数 加 密 后 的 数 
PASSWORD(str) | 据 是 不 可 逆 的 。 其 经 常用 于 对 用 户 注册 的 密码 | 对 子 符 囊 mrsof 进行 密码 ， 其 语句 如 下: 
进行 加 密 处 理 select PASSWORD('mrsoft"): 
对 字符 串 st 进行 加 密 。 经 常用 于 对 普通 数据 | 使 用 MD50 函 数 对 mso 字符 串 进行 加 密 , 其 
MDS(str) 语句 如 下 : 


进行 加 密 


select MDS5(Cmrsoft): 


ENCODE(str.pswd_str) 


使 用 字符 串 pswd_str 来 加 密 字符 串 str。 加密 的 
结果 是 一 个 二 进 制 数 , 必须 使 用 BLOB 类 型 的 
字段 来 保存 它 


使 用 字符 串 mr 对 mrsoft 进行 加 密 处 理 ， 其 语 
句 如 下 : 
SelectLENCODECmrsoff mr): 


DECODE(crypt_str. 
pswd_str) 
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使 用 字符 串 pswd_str 来 为 crypt_str 解密 。 
crypt_str 是 通过 ENCODE(str.pswd_str) 加 密 后 
的 二 进 制 数据 。 字符 串 pswd_str 应 该 与 加 密 时 
的 字符 串 pswd_str 相同 


应 用 DECODE 函数 对 经 过 ENCODE 函数 加 密 


“| 的 字符 串 进行 解密 ， 其 语句 如 下 : 


select DECODE(ENCODE(Cmrsoff mr).mr): 
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11.7.1 加 密 函 数 PASSWORD(str) 


PASSWORD(str) 函 数 可 以 对 字符 串 str 进行 加 密 。 一般 情况 下 ，PASSWORD(str) 函 数 主要 是 用 来 给 用 户 
的 密码 加 密 的 。 

例 11.26 下 面 使 用 PASSWORD(str) 函 数 为 字符 串 “abcd” 加 密 。 运 行 结果 如 图 11.26 所 示 。〔 实 例 位 
置 : 光盘 \TM\Instance\11\11.26) 

结果 显示 ， 字 符 串 “abcd ”加密 后 的 结果 是 *A154C2565E9E7F94BFC08A1lFE702624ED8EFDA 。 
PASSWORD(stD) 函 数 加 密 是 不 可 逆 的 。 


[3 技巧 了 PASSWORD(stD) 函 数 经 常用 来 给 密码 加 密 。 MySQL 用 户 需要 设置 密码 ， 用 户 不 能 将 未 加 密 的 
密码 直接 存储 到 MySQL 的 user 表 中 。 因 为 登录 MySQL 数据 库 时 ， 数 据 库 系统 会 将 你 输入 的 密码 先 通 
过 PASSWORD(stD) 函 数 加 密 ， 然 后 与 数据 库 中 的 密码 进行 比较 ， 匹 配 成 功 后 才 可 以 登录 。 


11.7.2 ”加 密 函 数 MD5(str) 


MD5(str) 函 数 可 以 对 字符 串 str 进行 加 密 。 主 要 对 普通 的 数据 进行 加 密 。 
例 11.27 下 面 使 用 MD5(str) 函 数 为 字符 串 “abcd” 加 密 。 运 行 结果 如 图 11.27 所 示 。〔 实 例 位 置 : 光 
盘 \TMNInstance\ll\11.27) 


图 11.26 加 密 函 数 PASSWORD(str) 图 11.27 加 密 函 数 MD5(str) 
结果 显示 ， 字 符 串 abcd 的 MD5 值 为 e2fc714c4727ee9395f324cd2e7f331f。 


11.8 其 他 函数 


MySQL 中 除了 上 述 内 置 函 数 以 外 ， 还 包含 很 多 函数 。 例 如 ， 数 字 格 式 化 函数 FORMAT(x,n)，IP 地 址 
与 数字 的 转换 函数 INET_ATON(ip), 还 有 加 锁 函 数 GET_LOCT(name,time)、 解 锁 函 数 RELEASE_LOCK(name) 
等 。 在 表 11.8 中 罗列 了 MySQL 中 支持 的 其 他 函数 。 


表 11.8 MySQL 的 其 他 函数 


函数 作 用 
FORMAT(x.n) | 将 数字 x 进行 格式 化 ， 将 x 保留 到 小 数 点 后 n 位 。 这 个 过 程 需要 进行 四 合 五 入 
ASCIIG ASCIIGS) 返 回 字 符 串 s 的 第 一 个 字符 的 ASCI 码 
BING) BIN(%) 返 回 x 的 二 进 制 编码 
HEX(x) HEX(x) 返 回 x 的 十 六 进 制 编码 
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续 表 
函数 作 用 
OCT(x) OCT(x) 返 回 x 的 八进制 编码 
CONVCfLP) CONVCflLP) 将 x 从 旭 进 制 数 变 成 亿 进 制 数 
INET ATON(P) INET_ATON(IP) 函 数 可 以 将 人 P 地 址 转换 为 数字 表示 
INET NTOA(N) INET_NTOA(N) 函 数 可 以 将 数字 n 转换 成 人 P 的 形式 


GET_LOCT(name,time) 函 数 定义 一 个 名 称 为 name、 持 续 时 间 长 度 为 time 秒 的 锁 。 
锁定 成 功 ， 返 回 1; 如 果 尝 试 超时 ， 返 回 0， 如 果 遇 到 错误 ， 返 回 NULL 
RELEASE_LOCK(name) 函 数 解除 名 称 为 name 的 锁 。 如 果 解 锁 成 功 ， 返 回 1; 
如 果 尝 试 超时 ， 返 回 0， 如 果 解 锁 失 败 ， 返 回 NULL 

IS_ FREE LOCKCname) 函 数 判断 是 否 使 用 名 为 name 的 锁 。 如 果 使 用 ， 返 回 0; 


GET LOCT(name.time) 


RELEASE LOCK(name) 


IS FREE LOCK(name) 


否则 返回 1 
ke Pe 将 表达 式 expr 重复 执行 count 次 ， 然 后 返回 执行 时 间 。 该 函数 可 以 用 来 判断 
四 MySQL 处 理 表达 式 的 速度 
CONVERT(s USING cs) 将 字符 串 s 的 字符 集 变 成 cs 


将 x 变 成 type 类 型 ， 这 两 个 函数 只 对 BINARY、CHAR、DATE、DATETIME、 
CAST(x AS type), CONVERT(x, type) | TIME、SIGNED INTEGER、UNSIGNED INTEGER 这 些 类 型 起 作用 。 但 两 种 方 
法 只 是 改变 了 输出 值 的 数据 类 型 ， 并 没有 改变 表 中 字段 的 类 型 


11.8.1 格式 化 函数 FORMAT(x,n) 


FORMAT(x,n) 函 数 可 以 将 数字 x 进行 格式 化 ， 将 x 保留 到 小 数 点 后 n 位 。 这 个 过 程 需要 进行 四 舍 五 入 。 
例如 ，FORMAT(2.356,2) 返 回 的 结果 将 会 是 2.36; FORMAT(2.353,2) 返 回 的 结果 将 会 是 2.35。 

例 11.28 下面 使 用 FORMAT(x, np) 函数 来 将 789.12545 和 876.4321 进行 格式 化 , 都 保留 到 小 数 点 3 位 。 
运行 结果 如 图 11.28 所 示 。 (实例 位 置 : 光盘 \TMNInstance\ll\11.28) 

结果 显示 ，789.12545 格式 化 后 的 结果 是 789.125; 876.4321 格式 化 后 的 结果 是 876.432。 这 个 数 都 保留 
到 小 数 点 后 3 位 ， 而 且 都 进行 了 四 舍 五 入 处 理 。 


和 ot FORMAT(x,n) 防 数 可 以 将 x 保留 到 小 数 点 后 nn 位。 在 格式 化 过 程 中 需要 进行 四 使 五 入 的 操作 。 
FORMAT(x.n) 函 数 与 ROUND(x.y) 函 数 返回 X 保 留 到 小 数 点 后 y 位 的 值 . 截断 时 需要 进行 四 使 五 入 处 理 。 


11.8.2 ”改变 字符 集 的 函数 


CONVERT(s USING es) 函数 将 字符 串 s 的 字符 集 变 成 cs。 
例 11.29 下 面 将 字符 串 “abecd” 的 字符 集 变 成 gbk。 运行 结果 如 图 11.29 所 示 。 (实例 位 置 ; 光盘 \TM 
Instance\11\11.29) 


图 11.28 格式 化 函数 FORMATO 图 11.29 ”改变 字符 集 的 函数 


第 11 章 MySQL 函数 之 选 


11.8.3 ”改变 字段 数据 类 型 的 函数 


CAST(x AS type) 和 CONVERT(x,type) 这 两 个 函数 将 x 变 成 type 类 型 .这 两 个 函数 只 对 BINARY 、CHAR、 
DATETIME 、 TIME 、SIGNED INTEGER 、UNSIGNED 
INTEGER 这 些 类 型 起 作用 。 但 两 种 方法 只 是 改变 了 输出 值 
得 数据 类 型 ， 并 没有 改变 表 中 字段 的 类 型 。 

例 11.30 下面 表 中 的 times 字段 为 DATETIME 类 型 ， 
将 其 变 为 DATE 类 型 , 或 者 TIME 类 型 .运行 结果 如 图 11.30 
所 示 。 (实例 位 置 ， 光盘 \TM\Instance\11\11.30) 

结果 显示 , times 字段 原来 的 取 值 是 2012-11-16 11:03:46, 这 是 DATETIME 类 型 ; CAST(times AS DATE) 
返回 的 结 果 是 2012-11-16, 这 说 明 类 型 已 经 变 成 了 DATE 型 ;CONVERT(times, TIME) 返 回 的 结果 是 11:03:46， 

这 说 明 类 型 已 经 变 成 了 TIME 类 型 。 


图 11.30 ”改变 字段 数据 类 型 的 函数 


11.9 实 战 
句 4 视 频 讲解 : 光盘 \TM\Video\ 第 11 章 \ 实 战 .exe 


11.9.1 字符 串 函 数 的 使 用 


例 31 在 本 章 介 绍 了 常用 的 字符 串 函数 ， 当 返回 字符 串 “me” 在 字符 串 “You love me .He love me” 
中 第 一 次 出 现 的 位 置 。 对 于 字符 串 函数 中 的 LOCATE(s1,s) 和 POSITION(s1 IN s) 函 数 是 从 字符 串 s 中 获取 sl 
F 始 位 置 。 《实例 位 置 ， 光 盘 \TM\Instance\11\11.31) 


代码 如 下 : 
Select LOCATE(me',You love me.He love me."); 
Select POSITION(Cme' IN ‘You love me.He love me.’); 


运行 结果 如 图 11.31 所 示 。 


图 11.31 字符 串 函 数 的 使 用 


11.9.2 ”查看 当前 数据 库 版 本 号 


例 11.32 用 VERSIONO 函 数 返回 数据 库 的 版 本 号 ，connection_ id0 函 数 返 回 服务 器 的 连接 数 ， 也 就 是 
到 现在 为 止 MySQL 服务 的 连接 次 数 : database0 和 schema0 函 数 返 回 当前 数据 库 名 。 (实例 位 置 : 光盘 \TM 
Instance\11\11.32) 
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代码 如 下 : 
Select VERSION(),DATABASE(),USER(),CONNECTION_ID(); 
运行 结果 如 图 11.32 所 示 。 


11.9.3 生成 3 个 1~100 之 间 的 随机 整数 


例 11.33 使 用 ROUND(x) 函 数 生成 一 个 与 数 x 最 接近 的 整数 ， 当 然 ， 也 可 以 使 用 FLOOR(x) 函 数 来 生 


成 一 个 小 于 或 者 等 于 x 的 最 大 整数 。RAND0 函 数 产生 的 是 随机 数 ， 所 以 每 次 执行 的 结果 都 会 是 不 一 样 的 。 
(实例 位 置 ， 光盘 \TM\Instance\11\11.33) 
代码 如 下 : 
Select ROUND(RAND()*100),FLOOR(RAND!()*100),CEILING(RAND()*100); 
运行 结果 如 图 11.33 所 示 


图 11.32 查看 当前 数据 库 版 本 号 图 11.33 生成 3 个 1~100 之 间 的 随机 整数 


11.9.4 数字 函数 的 使 用 


例 11.34 将 数字 1.98752895 保留 到 小 数 点 后 4 位 。 实例 位 置 : 光盘 \TM\Instance\11\11.34) 
代码 如 下 : 

select TRUNCATE(1.98752895,4); 

运行 结果 如 图 11.34 所 示 。 


11.9.5 ”加 密 函数 的 使 用 


例 11.35 使 用 MDS5 对 字符 串 “mrsoft” 进 行 加密 的 值 。〈 实 例 位置 ， 光盘 \TM\Instance\11\11.35) 
代码 如 下 : 

SELECT MD5(mrsoft); 

运行 结果 如 图 11.35 所 示 。 


图 11.34 数学 函数 的 使 用 图 11.35 加 密 函 数 的 使 用 
11.10 小 结 
本 章 介绍 了 MySQL 数据 库 提 供 的 内 部 函数 。 这 些 函 数 包 括 数 学 函数 、 字 符 串 函数 、 日 期 和 时 间 函 数 、 
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条 件 判 断 函数 、 系 统 信息 函数 和 加 密 函 数 等 。 字 符 串 和 时 间 函 数 是 本 章 重 点 介绍 的 内 容 。 条 件 判断 函数 是 
本 章 的 难点 ， 因 为 条 件 判 断 函数 涉及 很 多 条 件 判断 和 跳 转 的 语句 。 这 些 函 数 通 常 与 SELECT 语句 一 起 使 用 ， 
用 来 方便 用 户 的 查询 。 同 时 INSERT、UPDATE、DELECT 语句 和 条 件 表达 式 也 可 以 使 用 这 些 函 数 。 读 者 一 
定 要 上 机 实际 操作 这 些 函 数 ， 这 样 可 以 对 函数 了 解 得 更 加 透彻 。 


11.11 学 习 成 果 检 验 


1. 计算 3 的 2 次 方 。 (实例 位 置 光盘 \TM\Instance\11\11.36) 
2. 将 字符 串 “MRBCCD ”全 部 改 为 小 写 。《〈 实 例 位置 : 光盘 \TM\Instance\11\11.37) 
3. 将 字符 串 “mrbced” 逆 序 输出 。〈 实 例 位 置 : 光盘 \TMNInstance\ll\11.38) 
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(名! 视频 讲解 : 37 分 钟 ) 


MySQL 中 提供 了 很 多 操作 语句 , 通过 这 些 语句 可 以 对 数据 库 、 数 
据 表 ， 以 及 数据 表 中 的 数据 等 进行 操作 。 本 章 将 详细 介绍 My5QL 的 
语句 及 其 应 用 ， 包 括 创 建 、 查 看 、 选 择 、 删 除数 据 库 ; 创建 、 修 改 、 
更 名 、 删 除数 据 表 ; 插入 、 浏 览 、 修 改 、 删 除 记录 和 数据 库 的 备份 和 
恢复 ， 这 些 是 程序 开发 人 员 必 须 掌握 的 内 容 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 


H 掌握 MySQL 数据 库 操作 


WI 掌握 MySQL 数据 表 操 作 
让 掌握 MySQL 语句 操作 
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12.1 MySQL 数据 库 操 作 


区 41 视频 讲解 : 光盘 \TM\Video\ 第 12 章 \MySQL 数据 库 操作 .exe 


启动 并 连接 MySQL 服务 器 后 , 即 可 对 MySQL 数据 库 进行 操作 , 操作 MySQL 数据 库 的 方法 非常 简单 。 
下 面 进行 详细 介绍 。 


12.1.1 创建 数据 库 CREATE DATABASE 


使 用 CREATE DATABASE 语句 可 以 轻松 创建 MySQL 数据 库 。 语 法 如 下 : 

CREATE DATABASE ”数据 库 名 ; 

在 创建 数据 库 时 ， 数 据 库 命名 有 以 下 几 项 规则 。 

加 ”不 能 与 其 他 数据 库 重 名 ， 否 则 将 发 生 错 误 。 

回 ”名 称 可 以 由 任意 字母 、 阿 拉 伯 数字 、 下 划 线 (_) 和 “$” 组 成 ， 可 以 使 用 上 述 的 任意 字符 开头 ， 

但 不 能 使 用 单独 的 数字 ， 否 则 会 造成 数据 库 与 数值 相 混淆 。 

回 名称 最 长 可 为 64 个 字符 ， 而 别名 最 长 可 达 256 个 字符 。 

回 ”不 能 使 用 MySQL 关键 字 作为 数据 库 名 、 表 名 。 

在 默认 情况 下 ，Windows 下 数据 库 名 、 表 名 的 大 小 不 敏感 的 ， 而 在 Linux 下 数据 库 名 、 表 名 的 大 小 
写 是 敏感 的 。 为 了 便于 数据 库 在 平台 间 进 行 移植 ， 建 议 读者 采用 小 写 来 定义 数据 库 名 和 表 名 。 

例 12.1 通过 CREATE DATABASE 语句 创建 一 个 名 称 为 db_admin 的 数据 库 ， 如 图 12.1 所 示 。 (实例 
位 置 ， 光盘 \TM\Instance\12\12.1) 


12.1.2 ”查看 数据 库 SHOW DATABASES 


成 功 创建 数据 库 后 ， 可 以 使 用 SHOW 命令 来 查看 MySQL 服务 器 中 的 所 有 数据 库 信 息 。 语 法 如 下 : 

SHOW DATABASES; 

例 12.2 在 12.1.1 节 中 创建 了 数据 库 db_admin, 下 面 使 用 SHOW DATABASES 语句 查看 MySQL 服务 
器 中 的 所 有 数据 库 名 称 ， 如 图 12.2 所 示 。 (实例 位 置 : 光盘 \TMNInstance\12\12.2) 


@ 查看 所 有 数据 库 


数据 库 列表 


图 12.1 创建 MySQL 数据 库 图 12.2 查看 数据 库 
从 图 12.2 运行 的 结果 可 以 看 出 ，MySQL 服务 器 中 有 4 个 数据 库 。 


12.1.3 ”选择 数据 库 USE DATABASE 


在 上 面 的 讲解 中 ， 虽 然 成 功 创建 了 数据 库 ， 但 并 不 表示 当前 就 在 操作 数据 库 db_admin。 可 以 使 用 USE 
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语句 选择 一 个 数据 库 ， 使 其 成 为 当前 默认 数据 库 。 语 法 如 下 : 


USE 数据库 名 ; 
例 12.3 选择 名 称 为 db_admin 的 数据 库 ， 设 置 其 为 当前 默认 的 数据 库 ， 如 图 12.3 所 示 。〔 实 例 位 置 : 
光盘 \TM\Instance\12\12.3) 


12.1.4 ”删除 数据 库 DROP DATABASE 


删除 数据 库 的 操作 可 以 使 用 DROP DATABASE 语句 。 语 法 如 下 : 
DROP DATABASE ”数据 库 名 ; 


和 注意 六 又 订 的 操作 应 该 证 质 便 用， 一 瑟 拓 生计 打 作 ， 直 所 库 的 所 有 站 和 和 台 所 部 人 被 天 从 
有 恢复 的 可 能 ， 除 非 数据 库 有 备份 


例 12.4 通过 DROP DATABASE 语句 删除 名 称 为 db_admin 的 数据 库 ， 如 图 12.4 所 示 。 《实例 位 置 ; 
光盘 \TM\Instance\12\12.4) 


nin; 
@ 选择 指定 的 数据 库 


四 成 功 地 选择 了 该 数据 库 


图 12.3 选择 数据 库 图 12.4 删除 数据 库 


12.2 MySQL 数据 表 操 作 


著 g 视频 讲解 : 光盘 \TM\Video\ 第 12 章 \MySQL 数据 表 操 作 .exe 

在 对 MySQL 数据 表 进 行 操作 前 ， 必 须 首先 使 用 USE 语句 选择 数据 库 ， 才 可 在 指定 的 数据 库 中 对 数据 
表 进 行 操作 ， 如 创建 数据 表 、 修 改 表 结 构 、 更 改 数据 表 名 或 删除 数据 表 等 ， 否 则 是 无 法 对 数据 表 进 行 操作 
的 。 下 面 分 别 详细 介绍 对 数据 表 的 操作 方法 。 


12.2.1 创建 数据 表 CREATE TABLE 


创建 数据 表 使 用 CREATE TABLE 语句 。 语 法 如 下 : 
CREATE [TEMPORARY] TABLE [IF NOT EXISTS] 数据 表 名 
[(create_definition,...)][table_options] [select_statement] 
CREATE TABLE 语句 的 参数 说 明 如 表 12.1 所 示 。 


表 12.1 CREATE TABLE 语句 的 参数 说 明 


关键 字 说 明 
TEMPORARY | 如 果 使 用 该 关键 字 ， 表 示 创 建 一 个 临时 表 
IFNOTEXISTS | 该 关键 字 用 于 避免 表 存 在 时 MySQL 报告 的 错误 
create definition 表 的 列 属性 部 分 。MySQL 要 求 在 创建 表 时 ， 表 要 至 少 包 含 一 列 
table_options 表 的 一 些 特性 参数 
select_statement SELECT 语句 描述 部 分 ， 用 它 可 以 快速 地 创建 表 
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下 面 介绍 列 属性 create_definition 部 分 ， 每 一 列 定义 的 具体 格式 如 下 : 
col_name type [NOT NULL | NULL] [DEFAULT default_value] [AUTO_INCREMENT] 
[PRIMARY KEY ] [reference_definition] 


属性 create_definition 的 参数 说 明 如 表 12.2 所 示 。 
表 12.2 属性 create_definition 的 参数 说 明 
说 有明 


参数 


col name 
type 
NOT NULL | NULL 


字段 名 

字段 类 型 

指出 该 列 是 否 允 许 是 空 值 ， 系 统一 般 默 认 允 许 为 空 值 ， 所 以 当 不 允许 为 空 值 时 ， 必 须 使 用 
NOTNULL 

表示 默认 值 

表示 是 否 是 自动 编号 ， 每 个 表 只 能 有 一 个 AUTO_INCREMENT 列 ， 并 且 必 须 被 索引 
表示 是 否 为 主键 。 一 个 表 只 能 有 一 个 PRIMARY KEY。 如 果 表 中 没有 PRIMARY KEY, 而 
某 些 应 用 程序 需要 PRIMARY KEY，MySQL 将 返回 第 一 个 没有 任何 NULL 列 的 UNIQUE 
键 ， 作 为 PRIMARY KEY 

为 字段 添加 注释 


DEFAULT default_value 
AUTO INCREMENT 


PRIMARY KEY 


reference_definition 


以 上 是 创建 一 个 数据 表 的 一 些 基 础 知识 ， 看 起 来 十 分 复杂 ， 但 在 实际 应 用 中 使 用 最 基本 的 格式 创建 数 
据 表 即 可 。 具 体格 式 如 下 : 

create table table_name ( 列 名 1 属性 , 列 名 2 属性 .…); 

例 12.5 使 用 CREATE TABLE 语句 在 MySQL 数据 库 db_admin 中 创建 一 个 名 为 tb_admin 的 数据 表 ， 
该 表 包 括 id、user、password 和 createtime 等 字段 ， 如 图 12.5 所 示 。 (实例 位 置 : 光盘 \TM\Instance\12\12.5) 


12.2.2 ”查看 表 结 构 SHOW COLUMNS 或 DESCRIBE 


对 于 一 个 创建 成 功 的 数据 表 , 可 以 使 用 SHOW COLUMMNS 或 DESCRIBE 语句 查看 指定 数据 表 的 表 结 构 。 
下 面 分 别 对 这 两 个 语句 进行 介绍 。 

1. SHOW COLUMNS 语句 

SHOW COLUMNS 语句 的 语法 如 下 : 

SHOW [FULL] COLUMNS “FROM 数据 表 名 [FROM 数据 库 名 ]; 

或 写成 : 

SHOW [FULL] COLUMNS “FROM 数据 表 名 .数据 库 名 ; 

例 12.6 使 用 SHOW COLUMNS 语句 查看 数据 表 tb_admin 的 结构 ， 如 图 12.6 所 示 。〔 实 例 位 置 ， 光 
盘 \TMNInstance\12\12.6) 


炳 凡 溃 聘 诡 图 | 


图 12.5 创建 MySQL 数据 库 图 12.6 查看 表 结构 
2. DESCRIBE 语句 
DESCRIBE 语句 的 语法 如 下 : 
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DESCRIBE 数据 表 名 ; 

其 中 , DESCRIBE 可 以 简写 成 DESC。 在 查看 表 结构 时 , 也 可 以 只 列 出 某 一 列 的 信息 。 其 语法 格式 如 下 : 

DESCRIBE 数据 表 名 列 名 ; 

例 12.7 使 用 DESCRIBE 语句 的 简写 形式 查看 数据 表 tb_admin 中 的 某 一 列 信 息 , 如 图 12.7 所 示 。( 实 
例 位 置 ， 光盘 \TM\Instance\12\12.7) 


12.2.3 ”修改 表 结 构 ALTER TABLE 


修改 表 结 构 使 用 ALTER TABLE 语句 。 修 改 表 结构 指 增加 或 者 删除 字段 、 修 改 字段 名 称 或 者 字段 类 型 、 
设置 或 者 取消 主键 /外 键 、 设 置 或 者 取消 索引 以 及 修改 表 的 注释 等 。 语 法 如 下 : 
Alter[IGNORE] TABLE 数据 表 名 alter_spec[alter_spec]... 


全 注意 当 指定 IJGNORE 时 ， 如 果 出 现 重复 关键 的 行 ， 则 只 执行 一 行 ， 其 他 重复 的 行 被 删除 。 


其 中 ，alter_spec 子 句 定义 要 修改 的 内 容 ， 其 语法 如 下 : 
alter_specification: 


ADD [COLUMN] create_definition [FIRST | AFTER column_name ] /添加 新 字段 
ADD INDEX [index_name] (index_col_name,...) /添加 索引 名 称 
ADD PRIMARY KEY (index_col_name,...) // 添 加 主键 名 称 
ADD UNIQUE [index_name] (index_col_name,...) /添加 唯一 索引 
ALTER [COLUMN] col_name {SET DEFAULT literal | DROP DEFAULT} /修改 字段 名 称 
CHANGE [COLUMN] old_col_name create_definition /修改 字段 类 型 
MODIFY [COLUMN] create_definition /修改 子 句 定义 字段 
DROP [COLUMN] col_name /删除 字段 名 称 
DROP PRIMARY KEY /删除 主键 名 称 
DROP INDEX index_name /删除 索引 名 称 
RENAME [AS] new_tbl_name // 更 改 表 名 
table_options 


ALTER TABLE 语句 允许 指定 多 个 动作 ， 其 动作 间 使 用 逗号 分 隔 ， 每 个 动作 表示 对 表 的 一 个 修改 。 

例 12.8 添加 一 个 新 的 字段 email， 类 型 为 varchar(50)，not null， 将 字段 user 的 类 型 由 varchar(30) 改 为 
varchar(40)。《〔 实 例 位 置 ， 光盘 \TMNInstance\12\12.8) 

代码 如 下 : 

alter table tb_admin add email varchar(50) not null ,modify user varchar(40); 

在 命令 模式 下 的 运行 情况 如 图 12.8 所 示 。 


目 
数 
据 
表 
结 
构 


图 12.7 查看 表 的 某 一 列 信息 12.8 ”修改 表 结 构 


图 12.8 中 只 给 出 了 修改 user 字段 类 型 的 结果 ， 读 者 可 以 通过 语句 “mysql> show tb_admin;” 查 看 整个 
表 的 结构 ， 以 确认 email 字段 是 否 添 加 成 功 。 
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we 
说 8 
= 说 明 通过 alter 修改 表 列 ， 其 前 提 是 必须 将 表 中 数据 全 部 删除 ， 然 后 才 可 以 修改 表 列 。 


12.2.4 重 命名 表 RENAME TABLE 


重 命名 数据 表 使 用 RENAME TABLE 语句 。 其 语法 如 下 : 
RENAME TABLE 数据 表 名 1 To 数据 表 名 2 
Wal. 
说 明 该 语句 可 以 同时 对 多 个 数据 表 进 行 重 命名 ， 多 个 表 之 间 以 过 号 “” 分 陋 . 
例 12.9 重 命名 数据 表 tb_admin， 更 名 后 的 数据 表 为 tb_user， 如 图 12.9 所 示 。 (实例 位 置 : 光盘 \TM 
Instance\12\12.9) 


12.2.5 ”删除 表 DROP TABLE 
删除 数据 表 的 操作 很 简单 ， 同 删除 数据 库 的 操作 类 似 ， 使 用 DROP TABLE 语句 即 可 实现 。 语 法 如 下 : 


DROP TABLE 数据 表 名 ; 
例 12.10 ”删除 数据 表 tb_user， 如 图 12.10 所 示 。 (实例 位 置 ; 光盘 \TM\Instance\12\12.10) 


p table th_user; 
人 @ 删除 数据 表 tb_user 


nysql> » 上 @ 成 功 删除 数据 表 也 _user 


图 12.9 ”对 数据 表 进行 更 名 图 12.10 ”删除 数据 表 


入 
《sse 注意 删除 数据 表 的 操作 应 该 谨慎 使 用 。 一 旦 删除 了 数据 表 ，, 表 中 的 数据 将 会 全 部 清除 ,没有 备份 则 


在 删除 数据 表 的 过 程 中 ， 删 除 一 个 不 存在 的 表 将 会 产生 错误 ， 如 果 在 删除 语句 中 加 入 正 EXISTS 关键 
字 可 避免 此 错误 发 生 。 格 式 如 下 : 
DROP TABLE IF EXISTS 数据 表 名 ; 


12.3 MySQL 语句 操作 


多 中 视频 讲解 : 光盘 \IM\Video\ 第 12 章 \MySQL 语句 操作 .exe 
在 数据 表 中 插入 、 浏 览 、 修 改 和 删除 记录 可 以 在 MySQL 命令 行 中 使 用 SQL 语句 完成 ， 下 面 介绍 如 何 
在 MySQL 命令 行 中 执行 基本 的 SQL 语句 。 
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12.3.1 插入 记录 INSERT 


在 建立 一 个 空 的 数据 库 和 数据 表 时 ， 首 先 需要 考虑 的 是 如 何 向 数据 表 中 添加 数据 ， 该 操作 可 以 使 放 
INSERT 语句 来 完成 。 语 法 如 下 : 

insert into 数据 表 名 (column_name,column_name2, ..…. ) values (value1, value2, ... ) 

在 MySQL 中 , 一 次 可 以 同时 插入 多 行 记录 , 各 
行 记录 的 值 清单 在 values 关键 字 后 以 号 (,) 分 后， 轿 mts) 
而 标准 的 SQL 语句 一 次 只 能 插入 一 行 。 S rrr 

例 12.11 向 管理 员 信息 表 tb_admin 中 插入 一 “的 
条 数据 信息 ， 如 图 12.11 所 示 。 (实例 位 置 : 光盘 \ 、 
TM\Instance\12\12.11) 轩 92 生 家 入 记 条 


12.3.2 ”查询 数据 库 记 录 SELECT 


要 从 数据 库 中 把 数据 查询 出 来 ， 就 要 用 到 数据 查询 语句 SELECT。SELECT 语句 是 最 常用 的 查询 语句 ， 
其 使 用 方式 有 些 复杂 ， 但 功能 很 强大 。 其 语法 如 下 : 
select [distinct] [concat(col1,":",col2) as col] selection_list /要 查询 的 内 容 ， 选 择 哪 些 列 


from 数据 表 名 /指定 数据 表 

where primary_constraint // 查 询 时 需要 满足 的 条 件 ， 行 必须 满足 的 条 件 
group by grouping_columns // 如 何 对 结果 进行 分 组 

order by sorting_cloumns // 如 何 对 结果 进行 排序 

having secondary_constraint /查询 时 满足 的 第 二 条 件 

limit count // 限 定 输出 的 查询 结果 


这 就 是 SELECT 查询 语句 的 语法 ， 下 面 对 它 的 参数 进行 详细 讲解 。 

1. selection_list 

设置 查询 内 容 。 如 果 要 查询 表 中 所 有 列 ， 可 以 将 其 设置 为 “*”; 如 果 要 查询 表 中 某 一 列 或 多 列 ， 则 直 
接 输入 列 名 ， 并 以 “,” 为 分 隔 符 。 

例 12.12 ”查询 tb_mrbook 数据 表 中 所 有 列 和 user、pass 列 。 (实例 位 置 : 光盘 \TMNInstance\12\12.12) 


代码 如 下 : 
select * from tb_mrbook; /查询 数据 表 中 所 有 数据 
select user,pass from tb_mrbook; /查询 数据 表 中 user 和 pass 列 的 数据 


2. table_list (数据 表 名 ) 

指定 查询 的 数据 表 。 既 可 以 从 一 个 数据 表 中 查询 ， 也 可 以 从 多 个 数据 表 中 进行 查询 ， 多 个 数据 表 之 间 
用 “,” 进 行 分 隔 ， 并 且 通 过 WHERE 子 句 使 用 连接 运算 来 确定 表 之 间 的 联系 。 

例 12.13 从 tb_mrbook 和 tb_bookinfo 数据 表 中 查询 “bookname=' PHP 开发 实战 宝典 '” 的 作者 和 价格 。 
(实例 位 置 ， 光盘 \TM\Instance\12\12.13) 

其 代码 如 下 : 


select tb_mrbook.id,tb_mrbook.bookname, author,price from tb_mrbook,tb_bookinfo 
where tb_mrbook.bookname = tb_bookinfo.bookname and tb_bookinfo.bookname = 'PHP 开发 实战 宝典 “'; 


在 上 面 的 SQL 语句 中 ， 因 为 两 个 表 都 有 id 字段 和 bookname 字段 ， 为 了 告诉 服务 器 要 显示 的 是 哪个 表 
中 的 字段 信息 ， 要 加 上 前 级 。 语 法 如 下 : 
表 名 .字段 名 


他 
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“tb_mrbook.bookname = tb_bookinfo.bookname” 将 表 tb_mrbook 和 tb_bookinfo 连接 起 来 ， 叫 做 等 同 连 
接 : 如 果 不 使 用 “tb_mrbook.bookname = tb_ bookinfo bookname”， 那 么 产生 的 结果 将 是 两 个 表 的 笛 卡 儿 积 ， 
叫做 全 连接 。 

3. WHERE 条 件 语句 

在 使 用 查询 语句 时 ， 如 要 从 很 多 记录 中 查询 出 想 要 的 记录 ， 就 需要 一 个 查询 条 件 。 只 有 设 定 了 查询 条 
件 ， 查 询 才 有 实际 意义 。 设 定 查 询 条 件 应 用 的 是 WHERE 子 句 。 

WHERE 子 句 的 功能 非常 强大 ， 通 过 它 可 以 实现 很 多 复杂 的 条 件 查 询 。 在 使 用 WHERE 子 句 时 , 需要 使 
用 一 些 比 较 运算 符 ， 常 用 的 比较 运算 符 如 表 12.3 所 示 。 


表 12.3 ”常用 的 WHERE 子 句 比较 运算 符 


示 _ 例 
is not null Idis not null 
between Idbetweenl and 15 
| nm | Idin (3.4.5) 
| won | wa | Namenotin(shil) 


小 于 等 于 模式 匹配 Name like (shiok' 
| 等 | ia | nouike | 模式 史 配 Name not like (shi96) 
| wa | iaisnml | iesexp | 常规 表达 式 | Name 正则 表达 式 


表 12.3 中 列举 的 是 WHERE 子 句 常用 的 比较 运算 符 ， 示 例 中 的 这 是 记录 的 编号 ，name 是 表 中 的 用 户 名 。 

例 12.14 应 用 WHERE 子 句 ， 查 询 tb_mrbook 表 ， 条 件 是 type (类 别 ) 为 PHP 的 所 有 图 书 。〈 实 例 
位 置 ， 光盘 \TM\Instance\12\12.14) 

代码 如 下 : 

select * from tb_mrbook where type = 'PHP' 

4. GROUP BY 对 结果 分 组 

通过 GROUP BY 子 句 可 以 将 数据 划分 到 不 同 的 组 中 ， 实 现 对 记录 进行 分 组 查询 。 在 查询 时 ， 所 查询 的 
列 必须 包含 在 分 组 的 列 中 , 目的 是 使 查询 到 的 数据 没有 了 矛盾。 在 与 AVG0 或 SUMO 函 数 一 起 使 用 时 , GROUP 
BY 子 句 能 发 挥 出 最 大 作用 。 

例 12.15 查询 tb_mrbook 表 ， 按 照 type 进行 分 组 ， 求 每 类 图 书 的 平均 价格 。〔 实 例 位 置 ， 光 盘 \TM\ 
Instance\12\12.15) 

代码 如 下 : 

select bookname,avg(price),type from tb_mrbook group by type; 

5. DISTINCT 在 结果 中 去 除 重复 行 

使 用 DISTINCT 关键 字 ， 可 以 去 除 结果 中 重复 的 行 。 

例 12.16 查询 tb_mrbook 表 ， 并 在 结果 中 去 掉 类 型 字段 type 中 的 重复 数据 。〔 实 例 位 置 ， 光 盘 \TM\ 
Instance\12\12.16) 

代码 如 下 : 

select distinct type from tb_mrbook; 

6. ORDER BY 对 结果 排序 

使 用 ORDER BY 可 以 对 查询 的 结果 进行 升序 或 降序 排列 .在 默认 情况 下 , ORDER BY 按 升 序 输出 结果 。 
如 果 要 按 降序 排列 ， 可 以 使 用 DESC 来 实现 。 

在 对 含有 NULL 值 的 列 进行 排序 时 ， 如 果 是 按 升 序 排列 ，NULL 值 将 出 现在 最 前 面 ， 如 果 是 按 降序 排 


@ 
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列 ，NULL 值 将 出 现在 最 后 。 

例 12.17 查询 tb_mrbook 表 中 的 所 有 信息 , 按照 id 进行 降序 排列 , 并 且 只 显示 3 条 记录 。( 实 例 位置 : 
光盘 \TM\Instance\12\12.17) 

其 代码 如 下 : 

select * from tb_mrbook order by id desc limit 3; 

7. LIKE 模糊 查询 

LIKE 属于 较 常 用 的 比较 运算 符 ， 通 过 它 可 以 实现 模糊 查询 。 它 有 两 种 通配符 : “%” 和 下 划 线 (_) 。 
“%” 可 以 匹配 一 个 或 多 个 字符 ， 而 “_ ”只 匹配 一 个 字符 。 

例 12.18 查找 所 有 第 二 个 字母 是 “h” 的 图 书 。〈 实 例 位 置 : 光盘 \TM\Instance\12\12.18) 

select * from tb_mrbook where bookname like('_h%'); 


ys 3 RR 
8. CONCAT 联合 多 列 
使 用 CONCAT 函数 可 以 联合 多 个 字段 ， 构 成 一 个 总 的 字符 串 。 
例 12.19 把 tb_mrbook 表 中 的 书 名 (bookname) 和 价格 〈price) 合并 到 一 起 ， 构 成 一 个 新 的 字符 串 。 
《实例 位 置 ， 光 盘 \TM\Instance\12\12.19) 
代码 如 下 : 
select id,concat(bookname,":",price) as info,type from tb_mrbook; 
其 中 合并 后 的 字段 名 为 CONCAT 函数 形成 的 表达 式 “concat(bookname,":",price)”， 看 上 去 十 分 复杂 ， 
通过 AS 关键 字 给 合并 字段 取 一 个 别名 ， 这 样 看 上 去 就 很 清晰 。 
9. LIMIT 限定 结果 行 数 
LIMIT 子 句 可 以 对 查询 结果 的 记录 条 数 进 行 限定 ， 控 制 它 输出 的 行 数 。 
例 12.20 查询 tb_mrbook 表 ， 按 照 图书 价 格 降序 排列 ， 显 示 3 条 记录 。【〔 实 例 位 置 ， 光 盘 \TM\ 
Instance\12\12.20) 
代码 如 下 : 
select * from tb_mrbook order by price desc limit 3; 
使 用 LIMIT 还 可 以 从 查询 结果 的 中 间 部 分 取 值 。 首 先 要 定义 两 个 参数 ， 参 数 1 是 开始 读 取 的 第 一 条 记 
录 的 编号 (在 查询 结果 中 ， 第 一 个 结果 的 记录 编号 是 0， 而 不 是 1) ; 参数 2 是 要 查询 记录 的 个 数 。 
例 12.21 查询 tb_mrbook 表 , 从 编号 1 开始 ( 即 从 第 2 条 记录 ) 查询 4 个 记录 。( 实 例 位 置 : 光盘 \TM\ 
Instance\12\12.21) 
代码 如 下 : 
select * from tb_mrbook where id limit 1,4; 
10. 使 用 函数 和 表达 式 
在 MySQL 中 ， 还 可 以 使 用 表达 式 来 计算 各 列 的 值 ， 作 为 输出 结果 。 表 达 式 还 可 以 包含 一 些 函 数 。 
例 12.22 计算 tb_mrbook 表 中 各 类 图 书 的 总 价格 。〈 实 例 位置 : 光盘 \TM\Instance\12\12.22) 
代码 如 下 : 
select sum(price) as total,type from tb_mrbook group by type; 
在 对 MySQL 数据 库 进行 操作 时 ， 有 时 需要 对 数据 库 中 的 记录 进行 统计 ， 如 求 平均 值 、 最 小 值 、 最 大 值 
等 ， 这 时 可 以 使 用 MySQL 中 的 统计 函数 ， 常 用 统计 函数 如 表 12.4 所 示 。 


局 
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表 12.4 常用 统计 函数 

务 称 说 有明 
Avg (字段 名 ) 获取 指定 列 的 平均 值 
如 指定 了 一 个 字段 ， 则 会 统计 出 该 字段 中 的 非 空 记录 。 如 在 前 面 增 加 DISTINCT， 则 会 统计 不 同 
值 的 记录 ， 相 同 的 值 当 作 一 条 记录 。 如 使 用 COUNT(*) 则 统计 包含 空 值 的 所 有 记录 数 
Min 〈 字 段 名 ) 获取 指定 字段 的 最 小 值 
Max (字段 名 ) | 获取 指定 字段 的 最 大 值 
Std (字段 名 ) 指定 字段 的 标准 背离 值 
Stdtev 字段 名 〉 | 与 Std 相同 
Sum (字段 名 ) | 指定 字段 所 有 记录 的 总 合 


Count (字段 名 ) 


除了 使 用 函数 之 外 ， 还 可 以 使 用 算术 运算 符 、 字 符 串 运算 符 ， 以 及 逻辑 运算 符 来 构成 表达 式 。 
例 12.23 计算 图 书 打 八 折 之 后 的 价格 。〔 实 例 位 置 ， 光盘 \TM\Instance\12\12.23) 

代码 如 下 : 

select *, (price * 0.8) as '80%' from tb_mrbook:; 


12.3.3 ”修改 记录 UPDATE 


要 执行 修改 的 操作 ， 可 以 使 用 UPDATE 语句 。 其 语法 如 下 : 

update 数据 表 名 set column_name = new_value1,column_name2 = new_value2, ...where condition 

其 中 ，SET 子 句 指出 要 修改 的 列 和 它们 给 定 的 值 ， WHERE 子 句 是 可 选 的 ， 如 果 给 出 ， 将 指定 记录 中 哪 
行 应 该 被 更 新 ， 否 则 ， 所 有 的 记录 行 都 将 被 更 新 。 

例 12.24 将 管理 员 信息 表 tb_admin 中 用 户 名 为 tsoft 的 管理 员 密 码 111 修改 为 896552， 如 图 12.12 所 
示 。《〔 实 例 位 置 ， 光 盘 \TM\Instance\12\12.24) 


where user=’tsoft’; 


. 改 数 据 表 中 指定 条 件 的 记录 


! tmsoftBtmsoft-c 2812 -83-24 89:14:59 } 


| 人 @ 查看 修改 后 的 记录 信息 


图 12.12 修改 指定 条 件 的 记录 


和 5 注意 更 新 时 一 定 要 保证 WHERE 子 句 的 正确 性 ,一 旦 WHERE 子 名 出错 ， 将 会 破坏 所 有 改变 的 数据 。 


12.3.4 删除 记录 DELETE 


在 数据 库 中 ， 有 些 数据 已 经 失去 意义 或 者 出 现 错误 时 就 需要 将 它们 删除 ， 此 时 可 以 使 用 DELETE 语句 。 
其 语法 如 下 : 
delete from 数据 表 名 where condition 
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ea 
条 件 ， 将 按照 指定 的 条 件 进行 删除 。 


例 12.25 ”删除 管理 员 数 据 表 tb_admin 中 用 户 名 为 “小 欣 ” 的 记录 信息 ,如 图 12.13 所 示 。 (实例 位 置 ， 
光盘 \TM\Instance\12\12.25) 


EEE 有 删除 数据 表 tb admin 中 指定 条 件 的 记录 


图 12.13 ”删除 数据 表 中 指定 的 记录 


全 s 广 间 在 实际 应 用 中 ,执行 删除 操作 时 ,执行 删除 的 条 件 一 般 应 该 为 数据 的 id， 而 不 是 具体 某 个 字段 
值 ， 这 样 可 以 避免 一 些 不 必要 的 错误 发 生 。 


12.4 实 战 


铭 4 视频 讲解 : 光盘 \TM\Video\ 第 12 章 \ 视 频 12.4\ 实 战 .exe 


12.4.1 操作 teacher 表 


例 12.26 在 数据 库 中 创建 一 个 teacher 表 ， 并 且 将 teacher 表 的 name 字段 的 数据 类 型 改 为 varchar(30)， 
将 birthday 字段 的 位 置 改 到 sex 字段 的 前 面 。《〈 实 例 位 置 ， 光盘 \TMNInstance\12\12.26) 

代码 如 下 : 

Create table teacher(id int(4) not null primary key auto_increment, 

num int(10) not null , 

name varchar(20) not null, 

sex varchar(4) not null, 

birthday datetime, 

address varchar(50) 


Alter table teacher modify name varchar(30) not null; 


Alter table teacher modify birthday datetime after name; 
运行 效果 如 图 12.14 所 示 。 


12.4.2 ”登录 数据 库 系 统 


例 12.27 本 实例 在 命令 行 中 登录 MySQL 数据 库 管 理 系 统 ， 输 入 用 户 名 、 连 接 的 主机 、 密 码 ， 进 入 到 
mysql 命令 行 中 。【〔 实 例 位 置 ， 光盘 \TM\Instance\12\12.27) 


代码 如 下 : 
mysql -uroot ~h127.0.0.1 —p111 


运行 结果 如 图 12.15 所 示 。 
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图 12.14 操作 teacher 表 图 12.15 登录 数据 库 系统 


12.4.3” 读 取 MySQL 数据 库 中 数据 “PHP 语言 


例 12.28 读 取 数 据 库 中 的 数据 通过 mysql_query0 函 数 和 select 语句 查询 数据 , 使 用 mysql_fetch_assoc0 
函数 将 查询 结果 返回 到 数组 中 。 本 例 读 取 tb_news 表 中 的 新 闻 信息 。( 实 例 位 置 : 光盘 \TM\Instance\12\12.28) 


代码 如 下 : 
<?php 
”连接 数据 库 */ 
$conn=mysql_connect("localhost","root","111"); // 连 接 数据 库 服务 器 
mysql_select_db("db_database12",$conn); /| 选择 数据 库 
mysql_query("set names utf8"); // 设 置 编码 格式 
S$arr=mysql_query("select * from tb_news",$conn); /执行 查询 语句 
/使 用 while 语句 循环 mysql_fetch_assoc() 函 数 返 回 的 数组 %/ 
while($result=mysql_fetch_assoc($arr))f // 循 环 输出 查询 结果 
i 

<tr> 


<td height="25"><?php echo $resultrname];?> &nbsp;</td> ”// 输 出 新 闻 标 题 
<td height="25"><?php echo $resultrnews];?> </td> /| 输出 新 闻 内 容 
</tr> 
<?php 
} /结束 while 循环 


3> 
运行 效果 如 图 12.16 所 示 。 


新 闻 由 容 


最 全 面 的 开发 这 源 ,包括 实例 洋 通 库 . 技术 次 
| 逢 床 、 异 面 设 济 友 、 工 具 深 福 库 等 

| | 最 要 供 ， 于 发 者 种 衣 扶 到 入 适合 记 
用 不 尽 


ET 
后 可 本 入 个、 实现 仆人 的 六 和 委 源 ,有效 的 科学 各 人 


人 3 了 时 共生 和 二 新。 和 忆 扩 


图 12.16 浏览 新 闻 信 息 
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12.4.4 ”备份 和 恢复 MySQL 数据 库 〈Java 语言 ) 


例 12.29 ”本 实例 使 用 Mysqldump 命令 备份 数据 库 ， 运 行 结果 如 图 12.17 所 示 。 对 于 大 型 Web 项 目 来 
说 ， 具 有 良好 的 数据 备份 功能 是 非常 重要 的 ， 数 据 恢复 可 以 在 数据 遭 到 破坏 后 ， 将 备份 的 数据 恢复 到 数据 
库 系 统 中 ， 以 此 来 保证 系统 的 正常 运行 ， 以 免 数据 丢失 带 来 损失 。 数 据 库 恢复 的 运行 结果 如 图 12.18 所 示 。 
(实例 位 置 ， 光盘 \TM\Instance\12\12.29) 


i 数据 库 各 份 中 秒 报 库 诺 夏 
I SE 
dotibase0 Re ba) 
粳 入 备份 六 1 加 全 入 名 称 : CVs 3 
Eo 
法 径 因 要 蛋 夏 的 下 皇 床 : 
1 痢 的 又 首 格 z 区 :Yitass bak] 也 _databasec4 S 
[2 
图 12.17 数据 库 备份 的 运行 结果 12.18 数据 库 恢复 的 运行 结果 


数据 库 备 份 的 实现 过 程 如 下 。 
(1) 创建 UserDao 类 ， 定 义 数据 库 连 接 、 更 新 和 关闭 的 方法 ， 并 且 在 该 类 中 定义 数据 库 备份 的 方法 
mysqldumpO。 本 实例 中 连接 的 是 MySQL 的 系统 数据 库 information schema。 
(2) 创建 index.jsp 页 面 ， 读 取 information schema 数据 库 SCHEMATA 数据 表 中 的 数据 ; 创建 form 表 
单 ， 添 加 下 拉 列 表 ， 将 SCHEMATA 数据 表 中 的 数据 添加 到 下 拉 列 表 中 ;添加 文本 框 ， 设 置 备份 文件 的 存 
储 位 置 和 路 径 。 其 关键 代码 如 下 : 
<%@ page contentType="text/html; charset=gbk" language="java" import="java.sql.” errorPage="" %> 
<jsp:useBean id="dao" scope="request" class="com.pkh.dao.UserDao"/> 


<% 
ResultSet rs = dao.selectStatic("SELECT schema_name FROM SCHEMATA");// 执 行 查询 操作 , 获取 结果 集 
String path = request.getParameter("path"); // 获 取 备 份 文件 存储 的 位 置 
String dbase = request.getParameter("select"); // 获 取 指 定 备份 的 数据 库 


if(" 备 份 ".equals(request.getParameter("Submit"))X{ 
if (path != null && dbase != null) { 
if (dao.mysqldump(dbase,path)) { // 调 用 方法 ， 执 行 数 据 库 的 备份 
out.print("<script language='javascript'>alert(' 备 份 完成 ');</script>"); 
} 


} 
%> 
<form name="form1" method="post" action="> 
<select name="select"> 
<% 
while (rs.next()) { 
String database = rs.getString(1); 
out.printin("<option id=" + database + ">" + database + "</option>"); 


» 
dao.closeConnection(); 
%> 
</select> 
<input type="text" name="path"> 
<input type="submit" name="Submit" value=" 备 份 "> 
</form> 


@ 
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数据 库 恢复 的 关键 代码 如 下 : 
<%@ page contentType="text/html; charset=gbk" language="java" import="java.sql.*" errorPage="" %> 
<jsp:useBean id="dao" scope="request" class="com.pkh.dao.UserDao"/> 


<% 
ResultSet rs = dao.selectStatic("SELECT schema_name FROM SCHEMATA"); /执行 查询 语句 
String path = request.getParameter("path"); /获取 恢复 文件 存储 路 径 
String dbase = request.getParameter("select"); /获取 要 恢复 的 数据 库 


if ("恢复 ".equals(request.getParameter("Submit"))) { 
if (path != null && dbase != null) { 
if (dao.mysqlresume(dbase,path)) { /执行 恢复 操作 
out.print("<script language='javascript>alert(' 恢 复 完 成 ');</script>"); 
} 


} 
%> 
<form name="form1" method="post" action="> 
<input name="path" type="file" size="18"> 
<select name="select"> 
<% 
while (rs.next()) { 
String database = rs.getString(1); 
out.printin("<option id=" + database + ">" + database + "</option>"); 
} 
dao.closeConnection(); 
%> 
</select> 
<input type="submit" name="Submit" value=" 恢 复 "> 
</form> 


12.4.5 ”查看 表 详 细 结构 语句 SHOW CREATE TABLE 


例 12.30 下 面 用 SHOW CREATE TABLE 语句 查看 TEACHER 表 的 定义 。〔 实 例 位 置 ， 光盘 \IM\ 
Instance\12\12.30) 

代码 如 下 : 

SHOW CREATE TABLE TEACHER \G 

运行 结果 如 图 12.19 所 示 。 


oa 


图 12.19 查看 表 详 细 结构 


12;5. :村 结 


本 章 介绍 了 创建 表 和 库 、 查 看 表 结构 、 修 改 表 及 删除 表 和 库 的 方法 。 创 建 和 修改 表 和 库 的 内 容 比 较 重 
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要 ， 这 两 部 分 需要 不 断 的 练习 。 只 有 通过 实践 练习 ， 才 会 对 这 两 部 分 了 解 得 更 加 透彻 。 而 且 ， 这 两 部 分 很 
容易 出 现 语法 错误 ， 必 须 在 练习 中 掌握 正确 的 语法 规则 。 创 建 表 和 修改 表 后 一 定 要 查看 表 的 结构 ， 这 样 可 
以 确认 操作 是 否 正 确 。 删 除 表 时 一 定 要 特别 小 心 ， 因 为 删除 表 的 同时 会 删除 表 中 的 所 有 记录 。 


12.6 学习 成 果 检 验 


1. 如 何 查看 数据 库 ? 〈 实 例 位 置 : 光盘 \TM\Instance\12\12.31) 
2. 如 何 创建 数据 库 ? 《实例 位 置 ， 光盘 \TM\Instance\12\12.32) 
3. 如 何 把 原来 的 数据 表 重 命名 ? 《〈 实 例 位 置 : 光盘 \TMNInstance\12\12.33) 
4. 如 何 删除 数据 表 ? 实例 位 置 ， 光盘 \TM\Instance\12\12.34) 


RE 


MySQL 数据 查询 
(名! 视频 讲解 60 分 钟 ) 


数据 查询 是 指 从 数据 库 中 获取 所 需要 的 数据 。 数 据 查 询 是 数据 库 
操作 中 最 常用 ,也 是 最 重要 的 操作 。 用 户 可 以 根据 自己 对 数据 的 需求 
使 用 不 同 的 查询 方式 。 通 过 不 同 的 查询 方式 ， 可 以 获得 不 同 的 数据 。 
在 My5QL 中 是 使 用 SELECT 语句 来 查询 数据 的 。 本章 将 对 查询 语句 的 
基本 语法 、 在 单 表 上 查询 数据 、 使 用 聚合 函数 查询 数据 、 合 并 查询 结 
果 等 内 容 进 行 详细 地 讲解 ， 使 读者 轻松 了 解 查 询 数 据 的 语句 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

mm 了 解 MySQL 的 单 表 查询 

MH 了 解 使 用 聚合 图 数 实现 数据 查询 

ml 掌握 连接 查询 和 子 查询 

H 掌握 合并 查询 的 使 用 

WI 掌握 为 表 和 字段 取 别 名 的 用 法 

Mm 掌握 正则 表达 式 的 使 用 方法 
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13.1 基本 查询 语句 


区 | 视频 讲解 : 光盘 \TM\Video\ 第 13 章 \ 基 本 查询 语句 .exe 
SELECT 语句 是 最 常用 的 查询 语句 ， 它 的 使 用 方式 有 些 复杂 ， 但 功能 是 相当 强大 的 。SELECT 语句 的 基 


本 语法 如 下 : 
select selection_list /要 查询 的 内 容 ， 选 择 哪些 列 
from 数据 表 名 /指定 数据 表 
where primary_constraint /查询 时 需要 满足 的 条 件 ， 行 必须 满足 的 条 件 
group by grouping_columns // 如 何 对 结果 进行 分 组 
order by sorting_cloumns /如何 对 结果 进行 排序 
having secondary_constraint /查询 时 满足 的 第 二 条 件 
limit count /限定 输出 的 查询 结果 


其 中 使 用 的 子 句 将 在 后 面 逐 个 介绍 。 下 面 先 介绍 SELECT 语句 的 简单 应 用 。 
(1) 使 用 SELECT 语句 查询 一 个 数据 表 
使 用 SELECT 语句 时 ， 首 先 要 确定 所 要 查询 的 列 。“*” 代 表 所 有 的 列 ， 例 如 ， 查 询 db_database13 数 
据 库 user 表 中 的 所 有 数据 。 代 码 如 下 : 
mysql> use db_database13 
Database changed 
mysql> select * from user; 


sss + 

lid | user | lxdh |jtdz | 

十 一 一 十 -一 一 -十 - 十 - 十 

| 1|mr |12345678 | 长 春 市 | 

| 2|mrsoft |87654321 | 四 平市 | 

十 ~- 一 十 -一 一 -十 - 十 - 十 

2 rows in set (0.00 sec) 

这 是 查询 整个 表 中 所 有 列 的 操作 ， 还 可 以 针对 表 中 的 某 一 列 或 多 列 进行 查询 。 

(2) 查询 表 中 的 一 列 或 多 列 

针对 表 中 的 多 列 进行 查询 ， 只 要 在 select 后 面 指定 要 查询 的 列 名 即 可 ， 多 列 之 间 用 “,” 分 隔 。 例 如 ， 
查询 user 表 中 的 id 和 lxdh。 代 码 如 下 : 

mysql> select id , lxdh from user ; 

十 -一 -十 - 

lid | lxdh | 

十 -一 -十 十 

| 11123456781| 

| 2187654321 1 

十 -一 -十 - 

2 rows in set (0.00 sec) 


13.2 单 表 查 询 


铭 中 视频 讲解 : 光盘 \IM\Video\ 第 13 章 \ 单 表 查 询 .exe 
单 表 查询 是 指 从 一 张 表 中 查询 所 需要 的 数据 。 所 有 查询 操作 都 比较 简单 。 


局 
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13.2.1 查询 所 有 字段 


查询 所 有 字段 是 指 查 询 表 中 所 有 字段 的 数据 。 这 种 方式 可 以 将 表 中 所 有 字段 的 数据 都 查询 出 来 。 在 
MySQL 中 可 以 使 用 “* ”代表 所 有 的 列 ， 即 可 查 出 所 有 的 字段 。 语 法 格式 如 下 : 

SELECT * FROM 表 名 ; 

其 应 用 已 经 在 13.1 基本 查询 语句 中 领教 过 ， 这 里 不 再 袭 述 了 。 


13.2.2 ”查询 指定 字段 


查询 指定 字段 可 以 使 用 下 面 的 语法 格式 : 

SELECT 字段 名 FROM 表 名 ; 

如 果 是 查询 多 个 字段 ， 可 以 使 用 “,” 对 字段 进行 分 隔 。 

例 13.1 查询 db_database13 数据 库 tb_ login 表 中 “user” 和 “pwd” 两 个 字段 。 (实例 位 置 : 光盘 \TM\ 
Instance\13\13.1) 

SELECT 查询 语句 如 下 : 

SELECT user,pwd FROM tb_login: 

查询 结果 如 图 13.1 所 示 。 


13.2.3 ”查询 指定 数据 


如 果 要 从 很 多 记录 中 查询 出 指定 的 记录 , 那么 就 需要 一 个 查 
询 的 条 件 。 设 定 查询 条 件 应 用 的 是 WHERE 子 句 。 通 过 它 可 以 实 


图 13.1 查询 指定 字段 的 数据 
现 很 多 复杂 的 条 件 查询 。 在 使 用 WHERE 子 句 时 ， 需 要 使 用 一 些 比较 运算 符 来 确定 查询 的 条 件 。 常 用 比较 
运算 符 如 表 13.1 所 示 。 

表 13.1 比较 运算 符 


运算 符 


Is not null 


示 例 
Idis not null 
Idbetweenl and 15 


Between 


< Id<5 Idin (3.4.5) 
=> 大 于 等 于 | Id=>5 Notin Name not in (shili) 
= 小 于 等 于 “| 14<=5 Like Name like (‘shi%6") 


Name not like (‘shi%” 
E 则 表达 式 


不 等 于 Id!=5 Not like 


常规 表达 式 


表 13.1 中 列举 的 是 WHERE 子 句 常用 的 比较 运算 符 , 示例 中 的 id 是 
记录 的 编号 ，name 是 表 中 的 用 户 名 。 
例 13.2 应 用 where 子 句 , 查询 tb login 表 , 条 件 是 user (用 户 名 ) 


为 mr。 (实例 位 置 : 光盘 \TMNInstance\13\13.2) 
代码 如 下 : 图 13.2 查询 指定 数据 


select * from tb_login where user = 'mr; 


查询 结果 如 图 13.2 所 示 。 


Isnull Name 下 
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13.2.4 带 1N 关键 字 的 查询 


IN 关键 字 可 以 判断 某 个 字段 的 值 是 否 在 指定 的 集合 中 。 如 果 字 段 的 值 在 集合 中 ， 则 满足 查询 条 件 ， 该 
记录 将 被 查询 出 来 ， 如 果 不 在 集合 中 ， 则 不 满足 查询 条 件 。 其 语法 格式 如 下 : 

SELECT * FROM 表 名 WHERE 条 件 [NOT] IN( 元 素 1, 元 素 2,…, 元 素 n); 

NOT: 可 选 参数 ， 加 上 NOT 表示 不 在 集合 内 满足 条 件 。 

回 “ 元 素 : 集合 中 的 元 素 ， 各 元 素 之 间 用 逗号 隐 开 ， 字 符 型 元 素 需 要 加 上 单 引号 。 

例 13.3 应 用 IN 关键 字 查 询 tb_login 表 中 user 字段 为 mr 和 x 的 记录 .( 实 例 位 置 :光盘 \TMN\Instance\13\13.3) 

查询 语句 如 下 : 

i FROM tb_login WHERE user IN('mr','Ix'); 

查询 结果 如 图 13.3 所 示 。 

例 13.4 使 用 NOT IN 关键 字 查询 tb_login 表 中 user 字段 不 为 mr 和 lx 的 记录 。( 实 例 位 置 : 光盘 \TM\ 
Instance\13\13.4) 

查询 语句 如 下 

SELECT * FROM tb_login WHERE user NOT IN('mr','Ix'); 

查询 结果 如 图 13.4 所 示 。 


er NOT INC me’ 1x)s 


图 13.3 使 用 IN 关键 字 查询 图 13.4 使 用 NOT IN 关键 查询 


13.2.5 带 BETWEEN AND 的 范围 查询 


BETWEEN AND 关键 字 可 以 判断 某 个 字段 的 值 是 否 在 指定 的 范围 内 。 如 果 字 段 的 值 在 指定 范围 内 ， 则 
满足 查询 条 件 ， 该 记录 将 被 查询 出 来 ， 如 果 不 在 指定 范围 内 ， 则 不 满足 查询 条 件 。 其 语法 如 下 : 

SELECT * FROM 表 名 WHERE 条 件 [NOT] BETWEEN 取 值 1AND 取 值 2; 

回 NOT: 可 选 参数 ， 加 上 NOT 表示 不 在 指定 范围 内 满足 条 件 。 

回 取 值 1: 范围 的 起 始 值 。 

回 ” 取 值 2: 范围 的 终止 值 。 

例 13.5 查询 tb_login 表 中 id 值 在 5~7 之 间 的 数据 。 (实例 位 置 ， 光盘 \TM\Instance\13\13.5) 

查询 语句 如 下 : 

SELECT * FROM tb_login WHERE id BETWEEN 5 AND 7; 

查询 结果 如 图 13.5 所 示 。 

如 果 要 查询 tb_login 表 中 id 值 不 在 5 一 7 之 间 的 数据 ， 则 可 以 通过 NOT BETWEEN AND 来 完成 。 其 查 
询 语句 如 下 : 

SELECT * FROM tb_login WHERE id NOT BETWEEN 5 AND 7; 


13.2.6 带 LIKE 的 字符 匹配 查询 


LIKE 属于 较 常 用 的 比较 运算 符 ， 通 过 它 可 以 实现 模糊 查询 。 它 有 两 种 通配符 : “%” 和 下 划 线 “_”: 
“%” 可 以 匹配 一 个 或 多 个 字符 ， 可 以 代表 任意 长 度 的 字符 串 ， 长 度 可 以 为 0。 例 如 ，“ 明 % 技 ”表示 


@ 
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以 “ 明 ” 开 头 ， 以 “ 技 ” 结 尾 的 任意 长 度 的 字符 串 。 该 字符 串 可 以 代表 明日 科技 、 明 日 编程 科技 、 明 日 图 
书 科 技 等 字符 串 。 
“_” 只 匹配 一 个 字符 。 例 如 ，m_n 表示 以 m 开头 ， 以 n 结尾 的 3 个 字符 。 中 间 的 “_” 可 以 代表 任意 

一 个 字符 。 

例 13.6 查询 tb_login 表 中 user 字段 中 包含 mr 字符 串 的 数据 。( 实 例 位 置 : 光盘 \TM\Instance\13\13.6) 

查询 语句 如 下 : 

select * from tb_login where user like "%mr9%'; 

查询 结果 如 图 13.6 所 示 。 


和 


图 13.5 使 用 BETWEEN AND 关键 字 查询 图 13.6 模糊 查询 
13.2.7 用 IS NULL 关键 字 查询 空 


IS NULL 关键 字 可 以 用 来 判断 字段 的 值 是 否 为 空 值 (NULL) 。 如 果 字 段 的 值 是 空 值 ， 则 满足 查询 条 件 ， 
该 记录 将 被 查询 出 来 ， 如 果 字 段 的 值 不 是 空 值 ， 则 不 满足 查询 条 件 。 其 语法 格式 如 下 : 

IS [NOTI NULL 

其 中 ，NOT 是 可 选 参数 ， 加 上 NOT 表示 字段 不 是 空 值 时 满足 条 件 。 

例 13.7 下 面 使 用 IS NULL 关键 字 查 询 db_database13 数据 库 的 tb_book 表 中 row 字 段 的 值 为 空 的 记录 。 
(实例 位 置 ， 光 盘 \TM\Instance\13\13.7) 

查询 语句 如 下 : 

SELECT books,row FROM tb_book WHERE row IS NULL; 

查询 结果 如 图 13.7 所 示 。 


13.2.8 带 AND 的 多 条 件 查询 


AND 关键 字 可 以 用 来 联合 多 个 条 件 进 行 查询 。 使 用 AND 关键 字 时 ， 只 有 同时 满足 所 有 查询 条 件 的 记录 
会 被 查询 出 来 。 如果 不 满足 这 些 查 询 条 件 的 其 中 一 个 , 这样 的 记录 将 被 排除 掉 。 AND 关键 字 的 语法 格式 如 下 : 

select * from 数据 表 名 where 条 件 1and 条 件 2[...AND 条 件 表达 式 n]; 

AND 关键 字 连 接 两 个 条 件 表达 式 ， 可 以 同时 使 用 多 个 AND 关键 字 来 连接 多 个 条 件 表达 式 。 

例 13.8 下 面 查询 tb_login 表 中 user 字段 值 为 mr， 并 且 section 字段 值 为 PHP 的 记录 。〔 实 例 位 置 : 
光盘 \TM\Instance\13\13.8) 

查询 语句 如 下 : 

select * from tb_login where user='mr and section="php'; 

查询 结果 如 图 13.8 所 示 。 


| 


图 13.7 查询 tb_book 表 中 row 字段 值 为 空 的 记录 图 13.8 使 用 AND 关键 字 实现 多 条 件 查询 
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13.2.9 带 OR 的 多 条 件 查询 


OR 关键 字 也 可 以 用 来 联合 多 个 条 件 进行 查询 , 但 是 与 AND 关键 字 不 同 , OR 关键 字 只 要 满足 查询 条 件 
中 的 一 个 ， 那 么 此 记录 就 会 被 查询 出 来 ， 如 果 不 满足 这 些 查 询 条 件 中 的 任何 一 个 ， 这 样 的 记录 将 被 排除 掉 。 
OR 关键 字 的 语法 格式 如 下 : 

select * from 数据 表 名 where 条 件 1 OR 条 件 2[...OR 条 件 表达 式 n]; 

OR 可 以 用 来 连接 两 个 条 件 表达 式 。 而 且 ， 可 以 同时 使 用 多 个 OR 关键 字 连 接 多 个 条 件 表达 式 。 

例 13.9 下 面 查询 tb_login 表 中 section 字段 的 值 为 
PHP 或 者 “程序 开发 ”的 记录 。《〈 实 例 位 置 ， 光盘 \TM\ 
Instance\13\13.9) 

查询 语句 如 下 : 

select * from tb_login where section='php' or section=' 程 

序 开发 

查询 结果 如 图 13.9 所 示 。 


图 13.9 使 用 OR 关键 字 实现 多 条 件 查 询 


13.2.10 用 DISTINCT 关键 字 去 除 结果 中 的 重复 行 


使 用 DISTINCT 关键 字 可 以 去 除 查 询 结 果 中 的 重复 记录 。 语 法 格式 如 下 : 

select distinct 字段 名 from 表 名 ; 

例 13.10 下面 使 用 distinct 关键 字 去 除 tb_login 表 中 name 字段 中 的 重复 记录 。( 实 例 位 置 : 光盘 \TM\ 
Instance\13\13.10) 

查询 语句 如 下 : 

select distinct name from tb_login; 

查询 结果 如 图 13.10 所 示 。 去 除 重复 记录 前 的 name 字段 值 如 图 13.11 所 示 。 


> select distinct name from th_login; 


3 rows in set 8.00 sec) 


图 13.10 使 用 DISTINCT 关键 字 去 除 结果 中 的 重复 行 图 13.11 去 除 重复 记录 前 的 name 字段 值 


13.2.11 用 ORDER BY 关键 字 对 查询 结果 排序 


使 用 ORDER BY 关键 字 可 以 对 查询 的 结果 进行 升序 (ASC) 和 降序 (DESC) 排列 ， 在 默认 情况 下 ， 
ORDER BY 按 升序 输出 结果 。 如 果 要 按 降序 排列 可 以 使 用 DESC 来 实现 。 语 法 格式 如 下 : 

ORDER BY 字段 名 [ASCIDESC]; 

回 ASC: 表示 按 升 序 进行 排序 。 

加 ”DESC: 表示 按 降序 进行 排序 。 
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BS 

CA 
说 明 如 果 对 含有 NULL 值 的 列 进行 排序 时 ， 如 果 是 按 升序 排列 ，NULL 值 将 出 现在 最 前 面 ; 如 果 
是 按 降序 排列 ，NULL 值 将 出 现在 最 后 。 


例 13.11 查询 tb login 表 中 的 所 有 信息 ,按照 这 进行 降序 排列 。( 实 例 位 置 :光盘 \TMNInstance\13\13.11) 
查询 语句 如 下 : 
select * from tb_login order by id desc; 


查询 结果 如 图 13.12 所 示 。 


图 13.12 按 id 编 号 进行 降序 排列 


13.2.12 用 GROUP BY 关键 字 分 组 查询 


通过 GROUP BY 子 句 可 以 将 数据 划分 到 不 同 的 组 中 ， 实 现 对 记录 进行 分 组 查询 。 在 查询 时 ， 所 查询 的 
须 包 含 在 分 组 的 列 中 ， 目 的 是 使 查询 到 的 数据 没有 了 矛盾。 
1. 使 用 GROUP BY 关键 字 来 分 组 
单独 使 用 GROUP BY 关键 字 ， 查 询 结果 只 显示 每 组 的 一 条 记录 。 

例 13.12 使 用 GROUP BY 关键 字 对 tb_book 表 中 talk 字段 进行 分 组 查询 。 (实例 位 置 ， 光 盘 \TM 
Instance\13\13.12) 

查询 语句 如 下 : 

select id,books,talk from tb_book GROUP BY talk; 

查询 结果 如 图 13.13 所 示 。 

为 了 使 分 组 更 加 直观 明了 ， 下 面 查 询 tb_book 表 中 的 记录 ， 查 询 结 果 如 图 13.14 所 示 。 


列 必 


图 13.13 ”使 用 GROUP BY 关键 字 进 行 分 组 查询 图 13.14 ”tb_book 表 中 的 记录 


2. GROUP BY 关键 字 与 GROUP_CONCAT() 函 数 一 起 使 用 

使 用 GROUP BY 关键 字 和 GROUP_ CONCAT0 函 数 查 询 ， 可 以 将 每 个 组 中 的 所 有 字段 值 都 显示 出 来 。 

例 13.13 下面 使 用 GROUP BY 关键 字 和 GROUP_CONCATO 函 数 对 tb_book 表 中 的 takk 字 段 进行 分 组 
查询 。〔 实 例 位 置 ， 光盘 \TM\Instance\13\13.13) 

查询 语句 如 下 : 

select id,books,GROUP_CONCAT(talk) from tb_book GROUP BY talk; 

查询 结果 如 图 13.15 所 示 。 
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3. 按 多 个 字段 进行 分 组 
使 用 GROUP BY 关键 字 也 可 以 按 多 个 字段 进行 分 组 。 
例 13.14 ”下面 对 tb_book 表 中 的 user 和 sort 字段 进行 分 组 ， 分 组 过 程 中 ,按照 user 进行 分 组 。 (实例 
位 置 ， 光盘 \TM\Instance\13\13.14) 
查询 语句 如 下 : 
select id,books ,user from tb_book GROUP BY user; 
查询 结果 如 图 13.16 所 示 。 


图 13.15 ”使 用 GROUP BY 关键 字 与 GROUP_CONCATO 图 13.16 ”使 用 GROUP BY 关键 字 实 现 多 个 字段 分 组 
函数 进行 分 组 查询 


13.2.13 用 LIMIT 限制 查询 结果 的 数量 


查询 数据 时 ， 可 能 会 查询 出 很 多 记录 。 而 用 户 需 要 的 记录 可 能 只 是 很 少 一 部 分 。 这 样 就 需要 来 限制 查 
询 结果 的 数量 。LIMIT 是 MySQL 中 的 一 个 特殊 关键 字 。LIMIT 子 句 可 以 对 查询 结果 的 记录 条 数 进行 限定 ， 
控制 它 输出 的 行 数 。 下 面 通过 有 具体 实例 来 了 解 LIMIT 的 使 用 方法 。 

例 13.15 查询 tb login 表 中 ， 按 照 id 编号 进行 升序 排列 ， 显 示 前 3 条 记录 。〔 实 例 位 置 ， 光 盘 \TM\ 
Instance\13\13.15) 

查询 语句 如 下 : 

select * from tb_login order by id asc limit 3; 

查询 结果 如 图 13.17 所 示 。 

使 用 LIMIT 还 可 以 从 查询 结果 的 中 间 部 分 取 值 。 首 先 要 定义 两 个 参数 ， 参 数 1 是 开始 读 取 的 第 一 条 记 
录 的 编号 (在 查询 结果 中 ， 第 一 个 结果 的 记录 编号 是 0， 而 不 是 1) ; 参数 2 是 要 查询 记录 的 个 数 。 

例 13.16 查询 tb_login 表 中 , 按照 id 编号 进行 升序 排列 ， 从 编号 1 开始 ,查询 两 条 记录 。( 实 例 位 置 : 
光盘 \TM\Instance\13\13.16) 

查询 语句 如 下 : 

select * from tb_login where id order by id asc limit 1,2; 
查询 结果 如 图 13.18 所 示 。 


1 


图 13.17 使 用 LIMIT 关键 字 查询 指定 记录 数 图 13.18 使 用 LIMIT 关键 字 查询 指定 记录 


13.3 ”聚合 函数 查询 


句 4 视频 讲解 :光盘 \TM\Video\ 第 13 章 \ 聚 合 函 数 查询 .exe 
聚合 函数 的 最 大 特点 是 它们 根据 一 组 数据 求 出 一 个 值 。 聚 合 函 数 的 结果 值 只 根据 选 定 行 中 非 NULL 的 


@ 
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值 进行 计算 ，NULL 值 被 忽略 。 


13.3.1 COUNT() 函 数 


COUNTO 函 数 , 对 于 除 “*” 以 外 的 任何 参数 , 返回 所 选择 集合 中 非 NULL 值 的 行 的 数目 ; 对 于 参数 “*”， 
返回 选择 集合 中 所 有 行 的 数目 ， 包 含 NULL 值 的 行 。 没 有 WHERE 子 句 的 COUNT(*) 是 经 过 内 部 优化 的 ， 
能 够 快速 的 返回 表 中 所 有 的 记录 总 数 。 

例 13.17 下 面 使 用 count0 函 数 统计 tb_login 表 中 的 记录 数 。 (实例 位 置 : 光盘 \TM\Instance\13\13.17) 

查询 语句 如 下 : 

select count(*) from tb_login; 

查询 结果 如 图 13.19 所 示 。 结 果 显 示 ，tb_login 表 中 共有 4 条 
记录 。 


13.3.2 SUM() 函 数 县 
图 13.19 ”使 用 count0 函 数 统计 记录 数 

SUMO 函 数 可 以 求 出 表 中 某 个 字段 取 值 的 总 和 。 

例 13.18 使 用 SUMO 函 数 统计 tb_book 表 中 图 书 的 访问 量 字 段 (row) 的 总 和 。 在 查询 前 ， 先 来 查询 
-下 tb_ book 表 中 row 字段 的 值 ， 结 果 如 图 13.20 所 示 。 (实例 位 置 : 光盘 \TMNInstance\13\13.18) 

下 面 使 用 SUMO 函 数 来 查询 。 查 询 语句 如 下 : 

select sum(row) from tb_book; 

查询 结果 如 图 13.21 所 示 。 结 果 显 示 row 字段 的 总 和 为 96。 


图 13.20 tb_book 表 中 row 字段 的 值 图 13.21 使 用 SUMO 函 数 查 询 row 字段 值 的 总 和 
13.3.3 AVG() 函 数 


AVG0O 函 数 可 以 求 出 表 中 某 个 字段 取 值 的 平均 值 。 

例 13.19 下面 使 用 AVGO 函 数 求 tb_book 表 中 row 字段 值 的 平均 值 。 (实例 位 置 光盘 \TM\Instance\ 
13\13.19) 

查询 语句 如 下 : 

select AVG(row) from tb_book; 

查询 结果 如 图 13.22 所 示 。 


13.3.4 “MAX() 函 数 


MAXO 函 数 可 以 求 出 表 中 某 个 字段 取 值 的 最 大 值 。 
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例 13.20 ”下面 使 用 MAX0 函 数 查询 tb_book 表 中 row 字段 的 最 大 值 。( 实 例 位 置 : 光盘 \TMNInstance\13\13.20) 
查询 语句 如 下 : 

select MAX(row) from tb_book:; 

查询 结果 如 图 13.23 所 示 。 


图 13.22 使 用 AVGO 函 数 求 row 字段 值 的 平均 值 图 13.23 使 用 MAXO 函 数 求 row 字段 的 最 大 值 


下 面 来 看 一 下 tb_book 表 中 row 字段 的 所 有 值 ， 查 询 结果 如 图 13.24 所 示 。 结 果 显 示 row 字段 中 最 大 值 
为 953， 与 使 用 MAX 函数 查询 的 结果 一 致 。 


13.3.5_MIN() 函 数 


MINO 函 数 可 以 求 出 表 中 某 个 字段 取 值 的 最 小 值 。 

例 13.21 使 用 MINO 函 数 查 询 tb_book 表 中 row 字段 的 最 小 值 .( 实 例 位 置 :光盘 \TMNInstance\13\13.21) 
查询 语句 如 下 : 

select MIN(row) from tb_book; 

查询 结果 如 图 13.25 所 示 。 


图 13.24 tb_book 表 中 row 字段 的 所 有 值 图 13.25 使 用 MINO 函 数 求 row 字段 的 最 小 值 
13.4 连接 查询 


连接 是 把 不 同 表 的 记录 连 到 一 起 的 最 普遍 的 方法 。 一 种 错误 的 观念 认为 由 于 MySQL 的 简单 性 和 源 代码 
开放 性 ， 使 它 不 擅长 连接 。 这 种 观念 是 错误 的 。MySQL 从 一 开始 就 能 够 很 好 地 支持 连接 ， 现 在 还 以 支持 标 
准 的 SQL2 连接 语句 而 自 夺 ， 这 种 连接 语句 可 以 以 多 种 高 级 方法 来 组 合 表 记 录 。 


13.4.1 内 连接 查询 


铭 i 视 频 讲 解 : 光盘 \TMVVideo\ 第 13 章 \ 内 连接 查询 .exe 

内 连接 是 最 普遍 的 连接 类 型 ， 而 且 是 最 匀称 的 ， 因 为 它们 要 求 构成 连接 的 每 一 部 分 的 每 个 表 都 匹配 ， 
\ 匹 配 的 行将 被 排除 。 

内 连接 的 最 常见 的 例子 是 相等 连接 ， 也 就 是 连接 后 的 表 中 的 某 个 字段 与 每 个 表 中 的 都 相同 。 这 种 情况 
下 ， 最 后 的 结果 集 只 包含 参加 连接 的 表 中 与 指定 字段 相符 的 行 。 


这 
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例 13.22 下 面 有 两 个 表 , tb_login 用 户 信息 表 和 tb_book 图 书信 息 表 , 先 来 分 别 看 下 各 表 的 数据 , 图 13.26 
为 tb_login 表 的 数据 ， 图 13.27 为 tb_book 表 的 数据 。《〈 实 例 位 置 : 光盘 \TM\Instance\13\13.22) 


图 13.26 tb login 数据 表 图 13.27 tb_book 数据 表 


从 上 面 的 查询 结果 中 可 以 看 出 ， 在 两 个 表 中 存在 一 个 连接 user 字段 ， 它 在 两 个 表 中 是 等 同 的 ， 
tb_login 表 的 user 字段 与 tb_book 表 的 user 字段 相等 ， 因 此 可 以 创建 两 个 表 的 一 个 连接 。 查 询 语句 如 下 : 

select name,books from tb_login,tb_book where tb_login.user=tb_book.user; 

查询 结果 如 图 13.28 所 示 。 


13.4.2 ”外 连接 查询 


殴 il 视频 讲解 : 光盘 \TM\Video\ 第 13 章 \ 外 连接 查询 .exe 

与 内 连接 不 同 ， 外 连接 是 指使 用 OUTER JOIN 关键 字 将 两 个 表 连 接 起 来 。 外 连接 生成 的 结果 集 不 仅 包 
含 符合 连接 条 件 的 行 数据 ， 而 且 还 包括 左 表 〈 左 外 连接 时 的 表 ) 、 右 表 ( 右 外 连接 时 的 表 ) 或 两 边 连 接 表 
(全 外 连接 时 的 表 ) 中 所 有 的 数据 行 。 语 法 格式 如 下 : 

SELECT 字段 名 称 FROM 表 名 1 LEFTIRIGHT JOIN 表 名 2 ON 表 名 1. 字 段 名 1= 表 名 2. 属 性 名 2; 

外 连接 分 为 左 外 连接 (LEFT JOIN) 、 右 外 连接 (RIGHT JOIN) 和 全 外 连接 3 种 类 型 。 

1. 左 外 连接 

左 外 连接 (LEFT JOIN) 是 指 将 左 表 中 的 所 有 数据 分 别 与 右 表 中 的 每 条 数据 进行 连接 组 合 ， 返回 的 结果 
除 内 连接 的 数据 外 ， 还 包括 左 表 中 不 符合 条 件 的 数据 ， 并 在 右 表 的 相应 列 中 添加 NULL 值 。 


例 13.23 下 面 使 用 左 外 连接 查询 tb_login 表 和 tb_book 表 ， 通 过 user 字段 进行 连接 。 (实例 位 置 ， 光 
盘 \TM\Instance\13\13.23) 

查询 语句 如 下 : 

select section,tb_login.user,books,row from tb_login left join tb_book on tb_login.user=tb_book.user; 

查询 结果 如 图 13.29 所 示 。 


图 13.28 ”内 连接 查询 图 13.29 左 外 连接 查询 
结果 显示 ， 第 1 条 记录 的 books 和 row 字段 的 值 为 室 ， 这 是 因为 在 tb_book 表 中 并 不 存在 user 字段 为 
markj 的 值 。 
2. 右 外 连接 
右 外 连接 (RIGHT JOIN) 是 指 将 右 表 中 的 所 有 数据 分 别 与 左 表 中 的 每 条 数据 进行 连接 组 合 ， 返 回 的 结 
果 除 内 连接 的 数据 外 ， 还 包括 右 表 中 不 符合 条 件 的 数据 ， 并 在 左 表 的 相应 列 中 添加 NULL。 
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例 13.24 下 面 使 用 右 外 连接 查询 tb_book 表 和 tb_ login 表 ， 两 表 通 过 user 字段 连接 。 (实例 位 置 : 光 
盘 \TMNVInstance\13\13.24) 

查询 语句 如 下 : 

select section,tb_book.user,books,row from tb_book right join tb_login on tb_book.user=tb_login.user; 


查询 结果 如 图 13.30 所 示 。 
13.4.3 ”复合 条 件 连接 查询 


区 4 视频 讲解 : 光盘 \TM\Video\ 第 13 章 \ 复 合 条 件 连接 查询 .exe 
在 连接 查询 时 ， 也 可 以 增加 其 他 的 限制 条 件 。 通 过 多 个 条 件 的 复合 查询 ， 可 以 使 查询 结果 更 加 准确 。 
例 13.25 下 面 使 用 内 连接 查询 tb book 表 和 tb login 表 ， 并 且 tb_book 表 中 row 字段 值 必须 大 于 5。 
《实例 位 置 ， 光 盘 \TMNInstance\v13\13.25) 
查询 语句 如 下 : 
select section,tb_book.user,books,row from tb_book,tb_login where tb_book.user=tb_login.user and row>5; 
查询 结果 如 图 13.31 所 示 。 


图 13.30 右 外 连接 查询 图 13.31 复合 条 件 连 接 查 询 
13.5 子 查 询 


子 查询 就 是 SELECT 查询 是 另 一 个 查询 的 附属 。MySQL 4.1 可 以 嵌 套 多 个 查询 ， 在 外 面 一 层 的 查询 中 
使 用 里 面 一 层 查询 产生 的 结果 集 。 这 样 就 不 是 执行 两 个 〈 或 者 多 个 ) 独立 的 查询 ， 而 是 执行 包含 一 个 (或 
者 多 个 ) 子 查 询 的 单独 查询 。 

当 遇 到 这 样 的 多 层 查询 时 ，MySQL 从 最 内 层 的 查询 开始 ， 然 后 从 它 开始 向 外 向 上 移动 到 外 层 〈 主 ) 查 
询 ， 在 这 个 过 程 中 每 个 查询 产生 的 结果 集 都 被 赋 给 包围 它 的 父 查询 ， 接 着 这 个 父 查询 被 执行 ， 它 的 结果 也 
被 指定 给 它 的 父 查询 。 


查询 可 以 用 在 任何 可 以 使 用 表达 式 的 地 方 ， 它 必须 由 父 查询 包围 ， 而 且 ， 如 同 常规 的 SELECT 查询 ， 它 必 
须 包含 一 个 字段 列表 (这 是 一 个 单列 列表 ) 、 一 个 具有 一 个 或 者 多 个 表 名 字 的 FROM 子 句 以 及 可 选 的 
WHERE、HAVING 和 GROUP BY 子 句 。 


13.5.1 带 IN 关键 字 的 子 查询 


只 有 子 查询 返回 的 结果 列 包 含 一 个 值 时 ， 比 较 运 算 符 才 适 用 。 假 如 一 个 子 查询 返回 的 结果 集 是 值 的 列 
表 ， 这 时 比较 运算 符 就 必须 用 IN 运算 符 代 蔡 。 
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IN 运算 符 可 以 检测 结果 集中 是 否 存 在 某 个 特定 的 值 ， 如 果 检 测 成 功 就 执行 外 部 的 查询 。 

例 13.26 下面 查询 tb_login 表 中 的 记录 , 但 user 字段 值 必须 在 tb_book 表 中 的 user 字段 中 出 现 过 。( 实 
例 位 置 ， 光盘 \TM\Instance\13\13.26) 

查询 语句 如 下 : 

select * from tb_login where user in(select user from tb_book); 

在 查询 前 , 先 来 分 别 看 一 下 tb_login 和 tb_book 表 中 的 user 字段 值 ,以便 进行 对 比 , tb_login 表 中 的 user 
字段 值 如 图 13.32 所 示 ，tb_book 表 中 的 user 字段 值 如 图 13.33 所 示 。 


图 13.32 tb_login 表 中 的 user 字段 值 图 13.33 tb book 表 中 的 user 字段 值 
从 上 面 的 查询 结果 可 以 看 出 , 在 tb_book 表 的 user 字段 中 没有 出 现 mrkj 值 。 下 面 执行 带 IN 关键 字 的 子 
查询 语句 ， 查 询 结果 如 图 13.34 所 示 。 
查询 结果 只 查询 出 了 user 字段 值 为 lx 和 mrsoft 的 记录 ， 因 为 在 tb_book 表 的 user 字段 中 没有 出 现 mrkj 
的 值 。 


[a 技巧 NOT IN 关键 字 的 作用 与 IN 关键 字 刚 好 相反 。 在 本 例 中 ， 如 果 将 IN 换 为 NOT IN， 则 查询 结 
果 将 会 只 显示 一 条 user 字段 值 为 mrkj 的 记录 。 


13.5.2” 带 比较 运算 符 的 子 查询 


子 查 询 可 以 使 用 比较 运算 符 。 这 些 比较 运算 符 包括 =、!=、>、>=、<、<= 等 。 比 较 运算 符 在 子 查询 时 使 
用 得 非常 广泛 。 

例 13.27 下 面 查 询 图 书 访问 量 为 优秀 的 图 书 , 在 tb_row 表 中 将 图 书 访问 量 按 访 问 数 划分 等 级 , 如 图 13.35 
所 示 。《〔 实 例 位 置 ， 光 盘 \TM\Instance\13\13.27) 


图 13.34 使 用 IN 关键 字 实现 子 查询 图 13.35 查询 tb_row 表 中 的 数据 
从 结果 中 看 出 ， 当 访问 量 大 于 等 于 90 时 即 为 优秀 ， 下 面 再 来 查询 tb_book 图 书信 息 表 中 row 字段 的 值 ， 
如 图 13.36 所 示 。 
结果 显示 , 第 27 条 记录 的 访问 量 大 于 90。 下 面 使 用 比较 运算 符 的 子 查询 方式 来 查询 访问 量 为 优秀 的 图 


书信 息 。 查 询 语句 如 下 : 
select id,books,row from tb_book where row>=(select row from tb_row where id=1); 
查询 结果 如 图 13.37 所 示 。 
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图 13.36 查询 tb_book 表 中 row 字段 的 值 ” 图 13.37 使 用 比较 运算 符 的 子 查询 方式 来 查询 访问 量 为 优秀 的 图 书信 息 


13.5.3 带 EXISTS 关键 字 的 子 查询 


使 用 EXISTS 关键 字 时 ， 内 层 查 询 语句 不 返回 查询 的 记录 ， 而 是 返回 一 个 真 假 值 。 如 果 内 层 查询 语句 查 
询 到 满足 条 件 的 记录 ， 就 返回 一 个 真 值 (tue) ， 和 否则 ， 将 返回 一 个 假 值 (false) 。 当 返回 的 值 为 tue 时 ， 
外 层 查 询 语 句 将 进行 查询 ; 当 返 回 的 为 false 时 ， 外 层 查 询 语 句 不 进行 查询 或 者 查询 不 出 任何 记录 。 

例 13.28 下 面 使 用 子 查询 查询 tb_book 表 中 是 否 存在 id 值 为 27 的 记录 ， 如 果 存 在 则 查询 tb_row 表 中 
的 记录 ， 如 果 不 存在 则 不 执行 外 层 查 询 。 (实例 位 置 : 光盘 \TM\Instance\13\13.28) 

查询 语句 如 下 : 

select * from tb_row where exists (select * from tb_book where id=27); 

查询 结果 如 图 13.38 所 示 。 

因为 子 查询 tb_book 表 中 存在 id 值 为 27 的 记录 , 即 返 回 值 为 真 , 外 层 查 询 接 收 到 真 值 后 , 开始 执行 查询 。 

当 EXISTS 关键 与 其 他 查询 条 件 一 起 使 用 时 , 需要 使 用 AND 或 者 OR 来 连接 表达 式 与 EXISTS 关键 字 。 

例 13.29 如 果 tb_row 表 中 存在 name 值 为 优秀 的 记录 ， 则 查询 tb_book 表 中 row 字段 大 于 等 于 90 的 
记录 。《【〔 实 例 位置 ， 光盘 \TM\Instance\13\13.29) 

查询 语句 如 下 : 

select id,books,row from tb_book where row>=90 and exists(select * from tb_row where name=' 优 秀 '); 

查询 结果 如 图 13.39 所 示 。 


二 
而 


图 13.38 ”使 用 EXISTS 关键 字 的 子 查询 图 13.39 使 用 EXISTS 关键 字 查询 tb_book 表 中 
row 字段 大 于 等 于 90 的 记录 


多 技巧 NOT EXISTS 与 EXISTS 刚好 相反 。 使 用 NOT EXISTS 关键 字 ， 当 返回 的 值 是 true 时 ， 外 层 
查询 语句 不 执行 查询 ; 当 返 回 值 是 false 时 ， 人 外 层 查询 语句 将 执行 查询 。 


13.5.4” 带 ANY 关键 字 的 子 查询 


ANY 关键 字 表示 满足 其 中 任意 一 个 条 件 。 使 用 ANY 关键 字 时 ， 只 要 满足 内 层 查 询 语句 返回 的 结果 中 
的 任意 一 个 ， 就 可 以 通过 该 条 件 来 执行 外 层 查 询 语句 。 

例 13.30 查询 tb_ book 表 中 row 字段 的 值 小 于 tb_row 表 中 row 字段 最 小 值 的 记录 ， 首 先 查询 出 tb_row 表 
中 row 字段 的 值 ,然后 使 用 ANY 关键 字 (“<ANY ”表示 小 于 所 有 值 ) 判 断 。( 实 例 位 置 :光盘 \TMNInstance\13\13.30) 

查询 语句 如 下 : 


select books,row from tb_book where row<ANY (select row from tb_row); 


@ 
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查询 结果 如 图 13.40 所 示 。 


图 13.40 使 用 ANY 关键 字 实现 子 查询 


为 了 使 结果 更 加 直观 ， 下 面 分 别 查 询 tb_book 表 和 tb_row 表 中 的 row 字段 值 ， 查询 结果 如 图 13.41 和 
图 13.42 所 示 。 


图 13.41 tb_book 表 中 row 字段 的 值 图 13.42 tb row 表 中 row 字段 的 值 


结果 显示 ，tb_row 表 中 row 字段 的 最 小 值 为 50， 在 tb_book 表 中 row 字段 小 于 50 的 记录 有 1 条 , 与 带 
ANTY 关键 字 的 子 查询 结果 相同 。 


13.5.5” 带 ALL 关键 字 的 子 查询 


ALL 关键 字 表示 满足 所 有 条 件 。 使 用 ALL 关键 字 时 ， 只 有 满足 内 层 查 询 语句 返回 的 所 有 结果 ， 才 可 以 
执行 外 层 查 询 语句 。 

例 13.31 查询 tb_book 表 中 row 字段 的 值 大 于 tb_row 表 中 row 字段 最 大 值 的 记录 ， 首 先 使 用 子 查询 ， 
查询 出 tb_row 表 中 row 字段 的 值 ， 然 后 使 用 ALL 关键 字 (“>=ALL” 表 示 大 于 等 于 所 有 值 ) 判断 。 (实例 
位 置 光盘 \TM\Instance\13\13.31) 

查询 语句 如 下 : 

select books,row from tb_book where row>=ALL(select row from tb_row); 

查询 结果 如 图 13.43 所 示 。 


图 13.43 ”使 用 ALL 关键 字 实 现 子 查询 


为 了 使 结果 更 加 直观 ， 下 面 分 别 查询 tb_book 表 和 tb_row 表 中 的 row 字段 值 ,查询 结果 如 图 13.44 和 
图 13.45 所 示 。 


13.44 也 book 表 中 row 字段 的 值 
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结果 显示 ，tb_row 表 中 row 字段 的 最 大 值 为 90， 在 tb_book 表 中 row 字段 大 于 90 的 只 有 第 1 条 记录 ， 
与 带 ALL 关键 字 的 子 查询 结果 相同 。 
YL 
-说明 ANY 关键 字 和 ALL 关键 字 的 使 用 方式 是 一 样 的 ， 但 是 这 两 者 有 很 大 的 区 别 。 使 用 ANY 关键 
字 时 , 只 要 满足 内 层 查 询 语句 返回 的 结果 中 的 任何 一 个 , 就 可 以 通过 该 条 件 来 执行 外 层 查询 语句 ,而 ALL 
关键 字 则 需要 满足 内 层 查询 语句 返回 的 所 有 结果 ， 才 可 以 执行 外 层 查询 语句 。 


13.6 合并 查询 结果 


合并 查询 结果 是 将 多 个 SELECT 语句 的 查询 结果 合并 到 一 起 。 因 为 某 种 情况 下 ， 需 要 将 几 个 SELECT 
语句 查询 出 来 的 结果 合并 起 来 显示 。 合 并 查询 结果 使 用 UNION 和 UNION ALL 关键 字 。UNION 关键 字 是 
将 所 有 的 查询 结果 合并 到 一 起 , 然后 去 除 相同 记录 ; 而 UNION ALL 关键 字 则 只 是 简单 的 将 结果 合并 到 一 起 。 
下 面 分 别 介绍 这 两 种 合并 方法 。 

例 13.32 下面 查询 tb_book 表 和 tb_login 表 中 的 user 字段 ， 并 使 用 UNION 关键 字 合 并 查询 结果 。 在 
执行 查询 操作 前 ， 先 来 看 一 下 tb_book 表 和 tb_login 表 中 user 字段 的 值 ， 查 询 结 果 如 图 13.46 和 图 13.47 所 
示 。《〔 实 例 位 置 ， 光盘 \TM\Instance\13\13.32) 


图 13.46 tb_book 表 中 user 字段 的 值 图 13.47 tb_login 表 中 user 字段 的 值 


结果 显示 ， 在 tb_book 表 中 user 字段 的 值 有 3 种 ， 分 别 为 mr、mrsoft 和 Ix， 而 tb login 表 中 user 字段 
的 值 也 有 3 种 。 下 面 使 用 UNION 关键 字 合并 两 个 表 的 查询 结果 ， 查 询 语句 如 下 : 

select user from tb_book 

UNION 

select user from tb_login; 

查询 结果 如 图 13.48 所 示 。 结 果 显 示 ， 合 并 后 将 所 有 结果 合并 到 了 一 起 ， 并 去 除了 重复 值 。 

查询 tb_book 表 和 tb_login 表 中 的 user 字段 , 并 使 用 UNION ALL 关键 字 合 并 查询 结果 。 查询 语句 如 下 : 

select user from tb_book 

UNION ALL 

select user from tb_login; 

运行 结果 如 图 13.49 所 示 。 


图 13.48 ”使 用 UNION 关键 字 合 并 查询 结果 图 13.49 使 用 UNION ALL 关键 字 合 并 查询 结果 
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13.7 定义 表 和 字段 的 别名 


在 查询 时 ， 可 以 为 表 和 字段 取 一 个 别名 ， 这 个 别名 可 以 代 蔡 其 指定 的 表 和 字段 。 为 字段 和 表 取 别名 ， 
能 够 使 查询 更 加 方便 ， 而 且 可 以 使 查询 结果 以 更 加 合理 的 方式 显示 。 


13.7.1 为 表 取 别名 


当 表 的 名 称 特别 长 时 ， 在 查询 中 直接 使 用 表 名 很 不 方便 。 这 时 可 以 为 表 取 一 个 贴切 的 别名 。 

例 13.33 下 面 为 tb_program 表 取 别名 为 p， 然 后 查询 tb_program 表 中 talk 字段 值 为 php 的 记录 。 ( 实 
例 位置 ， 光 盘 \TM\Instance\13\13.33) 

查询 语句 如 下 : 

select * from tb_program p where p.talk='PHP'; 

tb_program p 表示 也 _program 表 的 别名 为 p;p.talk 表示 tb_program 表 中 的 talk 字段 ,查询 结果 如 图 13.50 
所 示 。 


13.7.2 为 字段 取 别 名 


当 查 询 数据 时 ，MySQL 会 显示 每 个 输出 列 的 名 词 。 默 认 情况 下 ， 显 示 的 列 名 是 创建 表 时 定义 的 列 名 。 
我 们 同样 可 以 为 这 个 列 取 一 个 别名 。 

MySQL 中 为 字段 取 别 名 的 基本 形式 如 下 : 

字段 名 [AS] 别名 

例 13.34 下 面 为 tb_login 表 中 的 section 和 name 字段 取 别 名 , 分 别 为 login_section 和 login_name。( 实 
例 位 置 ， 光盘 \TMNInstance\13\13.34) 

SQL 代码 如 下 : 

select section AS login_section,name AS login_name from tb_login; 

查询 结果 如 图 13.51 所 示 。 


图 13.50 为 表 取 别 名 . 和 为 字段 取 别 名 
13.8 使 用 正则 表达 式 查询 


正则 表达 式 是 用 某 种 模式 去 匹配 一 类 字符 串 的 一 个 方式 。 正 则 表达 式 的 查询 能 力 比 通 配 字符 的 查询 能 
力 更 强大 ， 而 且 更 灵活 。 下 面 详细 讲解 如 何 使 用 正则 表达 式 来 查询 。 

在 MySQL 中 ， 使 用 REGEXP 关键 字 来 匹配 查询 正则 表达 式 。 其 基本 形式 如 下 : 

字段 名 REGEXP ' 匹 配方 式 ' 


全 
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字段 名 : 表示 需要 查询 的 字段 名 称 。 
匹配 方式 ， 表示 以 哪 种 方式 来 进行 匹配 查询 。 


匹配 方式 参数 中 支持 的 模式 匹配 字符 如 表 13.2 所 示 。 


模式 字符 


含义 


表 13.2 ”正则 表达 式 的 模式 字符 
应 用 举例 


入 


匹配 以 特定 字符 或 字符 串 
开头 的 记录 


使 用 “^” 表 达 式 查询 tb_book 表 中 books 字段 以 字母 php 开头 的 记录 ， 语 
名 如 下 : 
Select books from tb_book Where books REGEXP ‘php'; 


匹配 以 特定 符 或 字符 串 结 
尾 的 记录 


使 用 “$” 表 达 式 查询 tb_book 表 中 books 字段 以 “模块 ”结尾 的 记录 ， 语 
句 如 下 : 
select books from tb book where books REGEXP ' 模 块 $': 


匹配 字符 串 的 任意 一 个 字 


使 用 “.” 表 达 式 来 查询 tb_book 表 中 books 字段 中 包含 P 字符 的 记录 ， 语 名 
如 下 : 


fA Select books from tb_book Where books REGEXP 'P.; 
匹配 字符 集合 中 的 任意 一 使 用 “[]” 表 达 式 来 查询 tb_book 表 中 books 字段 中 包含 PCA 字符 的 记录 ， 
[字符 集合 ] 个 字符 语句 如 下 : 
select books from tb_book Where books REGEXP '[PCAT'; 
[字符 集合 ] 匹配 除 字符 集合 以 外 的 任 | 查 询 tb_program 表 中 talk 字段 值 中 包含 c~z 字母 以 外 的 记录 ， 语 句 如 下 : 
”| 意 一 个 字符 select talk from tb_program where talk regexp '[^c-z]': 
匹配 S1、S2 和 S3 中 的 任 查询 本 表 中 books 字段 中 包含 php、c 或 者 java 字符 中 任意 一 个 字符 
S1|S2|S3 站 一 个 字符 让 的 记录 ， 语 句 如 下 : 
select books from tb books where books regexp 'phplcljava': 
匹配 多 个 该 符号 之 前 的 字 使 用 sy 表达 式 查 询 tb_book 表 中 books 字段 中 A 字符 前 出 现 过 了 字符 的 
符 ,包括 0 和 1 个 记录 ， 语 句 如 下 : 
Select books from tb_book Where books regexp J*A': 
匹配 多 个 该 符号 之 前 的 字 使 用 “+ 表达 式 来 查询 tb_book 表 中 books 字段 中 A 字符 前 面 至 少 出 现 过 
十 符 , 包括 1 个 一 个 J 字符 ， 语 句 如 下 : 
select books from tb_book Where books regexp ‘J+A': 
使 用 {N} 表 达 式 查询 tb_book 表 中 books 字段 中 连续 出 现 3 次 a 字符 的 记录 ， 
字符 串 {N} ”| 匹配 字符 串 出 现 N 次 语句 如 下 : 
select books from tb_book Where books regexp 'a{3} 
J 使 用 {MN} 表 达 式 查询 tb_book 表 中 books 字段 中 最 少 出 现 2 次 ， 最 多 出 现 
字符 中 {MN} | 到 字符 串 出 现 至 少 M | 4 次 a 字 符 的 记录 ， 语 句 如 下 : 


次 ， 最 多 N 次 


select books from tb book where books regexp 'a{2.4}': 


这 里 的 正则 表达 式 与 Java、PHP 等 编程 语言 中 的 正则 表达 式 基本 一 致 。 


13.8.1 


匹配 指定 字符 中 的 任意 一 个 


使 用 方 括号 〈[]) 可 以 将 需要 查询 的 字符 组 成 一 个 字符 集 。 只 要 记录 中 包含 方 括号 中 的 任意 字符 ， 该 记 
录 将 会 被 查询 出 来 。 例 如 ， 通 过 [abc] 可 以 查询 包含 a、b 和 < 等 3 个 字母 中 任何 一 个 的 记录 。 


例 13.35 ”下面 从 info 表 name 字段 中 查询 包含 e、e 和 o 3 个 字母 中 任意 一 个 的 记录 。 


@ 


(实例 位 置 : 光 
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盘 \TMNInstance\13\13.35) 
SQL 代码 如 下 : 
SELECT * FROM info WHERE name REGEXP '[ceo]': 
查询 结果 如 图 13.52 所 示 。 


13.8.2 使 用 “*” 和 “+” 来 匹配 多 个 字符 


正则 表达 式 中 ，“*” 和 “+” 都 可 以 匹配 多 个 该 符号 之 前 的 字 
符 。 但 是 ，“+” 至 少 表示 一 个 字符 ， 而 “* ”可 以 表示 0 个 字符 。 

例 13.36 下 面 从 info 表 name 字段 中 查询 字母 之 间 出 现 过 a 图 13.52 ”匹配 指定 字符 中 的 任意 一 个 
的 记录 。 (实例 位 置 : 光盘 \TM\Instance\13\13.36) 

SQL 代码 如 下 : 

SELECT * FROM info WHERE name REGEXP 'a*c'; 

查询 结果 如 图 13.53 所 示 。 

查询 结果 显示 ，Aric、Eric 和 Lucy 中 的 字母 之 前 并 没有 a。 因 为 “* ”可 以 表示 0 个 ， 所 以 “axc” 表 
示 字 母 c 之 前 有 0 个 或 者 多 个 a 出 现 。 上述 的 情况 都 是 属于 前 面 出 现 过 0 个 的 情况 。 如果 使 用 “+”, 其 SQL 
代码 如 下 : 

SELECT * FROM info WHERE name REGEXP 'a+c'; 

代码 执行 结果 如 图 13.54 所 示 。 


lect Fron info where none regoxp ox’s 


图 13.53 使 用 “*” 来 匹配 多 个 字符 图 13.54 使 用 “+” 来 匹配 多 个 字符 
查询 结果 只 有 一 条 。 只 有 Jack 是 刚好 字母 c 前 面 出 现 了 a。 因为 afc 表示 字母 前 面 至 少 有 一 个 字母 a。 


13.9 实 战 
区 4 视频 讲解 : 光盘 \TM\Video\ 第 13 章 \ 实 战 .exe 


13.9.1 使 用 聚合 函数 SUM 对 学 生成 绩 进行 汇总 


例 13.37 在 学 生成 绩 管理 系统 中 , 经 常 需要 求 出 班级 所 有 学 生 的 各 科 成 绩 总 和 ， 从 而 求 出 所 有 学 生 
各 科 平 均 成 绩 。 运 行 本 实例 ， 单 击 图 中 的 “对 学 生成 绩 汇总 ”按钮 即 可 将 班级 的 各 科学 生成 绩 统计 出 来 。 
(实例 位 置 ， 光盘 \TM\Instance\13\13.37) 


代码 如 下 : 

<?php 

$conn = mysql_connect("localhost","root","111") or die ("connect mysql false"”); /连接 MySQL 
mysql_select_db("db_database13",$conn) or die ("connect database false"); // 连 接 MySQL 数据 库 
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mysql_query("set names utf8"); /设置 编码 格式 
if$_GET[page] == "1"){ JWpage 等 于 1 时 
Srss = mysql_query("select *,sum(chinese+math+english) as a from tb_demo059 group by name order by 
a desc "); /返回 结果 集 
?> 


<table width="580px"><tr><td background="pic/head.JPG">ID</td><td background="pic/head.JPG"> 学 生 姓 名 
</td><td background="pic/head.JPG"> 语文 成 绩 </td><td background="pic/head.JPG"> 数学 成 绩 </td><td 
background="pic/head.JPG"> 英 语 成 绩 </td><td background="pic/head.JPG"> 总 成 绩 </td></tr> 
<?php 

while($rst = mysql_fetch_row($rss)X{ /1/ 循 环 输出 结果 
?> 
<tr><td background="pic/head.JPG"><?php echo S$rst[0];?></td><td background="pic/head.JPG"><?php echo 
S$rst[1];?></td><td background="pic/head.JPG"><?php echo Srst[2];?></td><td background="pic/head.JPG"> 
<?php echo S$rst[3];?></td><td background="pic/head.JPG"><?php echo Srst[4];?></td><td background= 
"pic/head.JPG"><?php echo $rst[5]?></td></tr> 
<?php 


} 


|! 


> 
运行 结果 如 图 13.55 所 示 。 


[车 全 用 吧 呈 因 效 恒 生成 绩 进 行 汇总 


13.9.2 ”查询 大 于 指定 条 件 的 记录 一 一 一 一 一 一 一 一 一 一 一 一 
例 13.38 在 图 书 管理 系统 中 有 时 需要 查询 图 书 价 ” | 一 一 和 @ 
格 大 于 其 他 图 书 的 详细 信息 。 运 行 本 实例 ， 首 先 在 图 中 图 13.55 ”对 学 生成 绩 汇总 


百 单 
息 查 找 出 来 。 (实例 位 置 : 光盘 \TM\Instance\13\13.38) 


的 文本 框 中 输入 查询 的 图 书 价格 应 大 于 的 数值 ， 多 
击 “ 查 看 ”按钮 即 可 将 图 书 表 中 大 于 该 数值 的 图 书信 


代码 如 下 : 
<?php 
if($_POST[sub']){ // 单 击 “ 查 看 ”按钮 
if($_POST[text1] == " ||$_POST[text1] == "输入 价格 "X{ /判断 文本 框 信息 
echo "<script>alert(' 请 输入 价格 ');</script>"; JavaScript 提示 
jelsef 
if(preg_match("/[*A-Za-z_1/",$_POST[text1])X{ // 正 则 表达 式 
$conn = mysql_connect("localhost","root","111") or die("connect mysql false"); 
mysql_select_db("db_database13",$conn) or die("connect database false"); 
/连接 MySQL 数据 库 
mysql_query("set names utf8"); // 设 置 编码 格式 
Srss = mysql_query("select * from tb_demo047 where price > $_POST[text1] order by 
price ); 


echo'<table width="580px"><tr><td background="pic/head.JPG">ID</td><td background= 
"pic/head.JPG"> 书 名 </td><td background="pic/head.JPG"> 价格 <jtd><td background="pic/head.JPG"> 日 期 
</td></tr>"; 

while($rst = mysql_fetch_array($rss))f // 循 环 输出 结果 
?> 


<tr><td background="pic/head.JPG"><?php echo Srst[0];?></td><td background="pic/head.JPG"> 
<?php echo S$rst[1];?></td><td background="pic/head.JPG"><?php echo Srst[2];?></td><td background= 
"pic/head.JPG"><?php echo S$rst[3];?></td></tr> 
<?php 


后 


} 
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jelse{ 
echo "<script>alert(' 不 是 正确 的 数值 ');</script>"; // 输 出 提示 
让 


} 


?> 
运行 结果 如 图 13.56 所 示 。 


13.9.3 ”使 用 比较 运算 符 进行 子 查询 


鹤 询 六 于 抱 定 委 符 的 记录 


例 13.39 从 computer stu 表 中 查询 获得 一 等 奖学金 的 
学 生 的 学 号 、 姓 名 和 分 数 。 各 个 等 级 的 奖学金 的 最 低 分 存储 图 13.56 查询 大 于 指定 条 件 的 记录 
在 score 表 中 。 (实例 位 置 ， 光盘 \TMNInstance\13\13.39) 

代码 如 下 : 

select id,name,score from computer_stu where score>=(select score from score where level=1); 
行 结果 如 图 13.57 所 示 。 


图 13.57 全 部 数据 图 13.58 ”使 用 比较 运算 符 进行 查询 


13.9.4 GROUP BY 与 HAVING 关键 字 


例 13.40 下 面 按 tb_student 表 的 sex 字段 进行 分 组 查询 ， 然 后 显示 记录 数 大 于 等 于 3 的 分 组 。〈 实 例 
位 置 : 光盘 \TM\Instance\13\13.40) 

代码 如 下 : 

SELECT sex,COUNT(sex) FROM tb_student GROUP BY sex HAVING COUNT(sex)>=3; 

运行 结果 如 图 13.59 所 示 。 


13.9.5 用 符号 “.” 来 替代 字符 串 中 的 任意 一 个 字符 


例 13.41 下 面 使 用 符号 “.” 来 蔡 代 字符 串 中 的 任意 一 个 字符 。( 实 例 位 置 : 光盘 \TM\Instance\13\13.41) 
代码 如 下 : 
SELECT * FROM aa WHERE name REGEXP *L..y$’; 
运行 结果 如 图 13.60 所 示 。 


PHP+MySQL 开发 实战 


图 13.39 GROUP BY 与 HAVING 关键 字 图 13.60 用 符号 “.” 来 替代 字符 串 中 任意 一 个 字 移 
13.10 小 结 


本 章 对 MySQL 数据 
学 知识 的 用 法 。 在 阅读 本 章 时 ， 读 者 应 该 重点 掌握 多 条 件 查询 、 连 接 查 询 、 子 查询 和 查询 结果 排序 ， 及 使 
用 正则 表达 式 来 查询 。 正 的 功能 很 强大 ， 使 用 起 来 很 灵活 。 和 希望 读者 能 够 阅读 有 关 正 则 表达 式 的 
书籍 ， 使 自己 对 正则 表达 式 了 解 得 更 加 透彻 。 


13.11 学 习 成 果 检 验 


.如 何 查 询 所 有 字段 ?答案 位 置 ， 光盘 \TMNInstance\13\13.42) 

.如 何 查询 带 like 的 字符 匹配 查询 ?答案 位 置 ， 光盘 \TM\Instance\13\13.43) 
.如 何 为 表 取 别 名 ? 〔 答 案 位 置 ， 光盘 \TMNInstance\13\13.44) 

.什么 是 子 查询 ?答案 位 置 ， 光盘 \TMNInstance\13\13.4S) 


和 wii 一 
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综合 实例 (二) 
(名 视频 讲解 : 3S 分 钟 ) 


以 构建 一 个 适合 于 中 小 型 企业 应 用 的 留言 本 为 目的 ， 全 面 介绍 留 
言 本 的 设计 思路 及 流程 。 在 介绍 留言 本 开发 过 程 的 同时 ， 对 PHP 中 一 
些 常 用 函数 及 编程 技巧 也 会 详细 介绍 。 通 过 本 章 的 讲解 ， 不 仅 可 以 对 
留言 本 的 开发 过 程 及 思路 有 个 全 面 地 带 担 ， 而 且 还 可 以 学 习 到 许多 
PHP 的 编程 技巧 ， 从 而 金 面 提高 个 人 的 基础 及 编程 能 力 。 留 言 本 的 开 
发 过 程 虽 然 较为 简单 , 却 涵盖 了 动态 网 站 开发 的 大 部 分 经 典 功 能 模块 ， 
所 以 只 要 就 悉 掌 担 本 章 的 内 容 ， 并 且 能 够 灵活 应 用 ， 就 可 以 开发 出 大 
型 的 网 上 论坛 、 博 客 以 及 其 他 一 些 网 络 中 较 流 行 的 网 站 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

NM 了解 留言 本 的 设计 思路 及 流程 

Wm 掌握 MySQL 数据 库 的 设计 方式 和 方法 

WI 掌握 PHP 连接 MySQL 数据 库 的 方法 

让 熟悉 运用 常见 的 SQL 查询 语句 

Wm 掌握 PHP 实现 数据 分 页 显示 的 方法 

WI 熟悉 对 留言 本 的 添加 、 查 看 和 编辑 
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14.1 留言 本 概述 


网 络 上 随处 可 见 各 种 各 样 的 留言 板 ， 一 个 网 页 甚至 一 条 新 闻 都 有 一 个 留言 板 支持 。 而 留言 本 是 留言 板 
的 一 个 纵向 延伸 ， 留 言 本 除 具有 留言 、 查 看 、 回 复 、 查 询 、 删 除 等 功能 外 ， 还 增加 了 管理 员 〈 也 称 为 版 主 ) 
管理 功能 。 留 言 本 被 广泛 应 用 于 小 规模 企业 ， 它 不 仅 为 企业 员工 提供 一 个 交流 的 平台 ， 也 成 为 企业 与 员工 
之 间 交 流 的 纽带 。 


14.2 系统 分 析 流 程 
句 1 视频 讲解 : 光盘 \TM\Video\ 第 14 章 \ 系 统 分 析 流程 .exe 


14.2.1 ”程序 业务 流程 


用 户 在 使 用 留言 本 进行 留言 时 ， 首 先 应 该 注册 ， 注 册 成 功 后 即 可 登录 本 站 ， 在 登录 本 留言 本 时 ， 系 统 
会 判断 用 户 的 级 别 ， 并 根据 用 户 的 级 别 为 其 分 配 不 同 的 权限 ， 非 本 留言 本 的 用 户 只 具有 查看 留言 内 容 的 权 
限 ， 普 通 注册 用 户 具有 添加 留言 、 编 辑 个 人 留言 和 删除 个 人 发 布 的 留言 以 及 回复 其 他 留言 的 权限 ， 而 管理 
员 除 具有 上 述 权 限 外 ， 还 具有 删除 全 部 留言 〈 即 留言 和 回复 ) 的 权限 以 及 对 自己 发 表 的 留言 进行 回复 的 权 
限 。 用 户 对 本 留言 本 进行 操作 的 业务 流程 如 图 14.1 所 示 。 


二 号 用 户主 册 信息 
上 退出 攻 录 
4 
和 用 户 信息 是 否 合法 
9 回 夏 久 言 
是 四 
2 
保存 用 户 汪 骨 信息 回复 其 他 留言 
了 发 表 留言 
用 户 入骨 记功 
i 天 党 录 信息 是 否 合法 
fs 


14.1 留言 本 业务 流程 图 


14.2.2 ”系统 预览 


留言 本 由 多 个 板块 组 成 ， 为 了 让 读者 对 本 留言 本 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 几 个 典型 功能 的 页 


@ 
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面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 


用 户 注册 信息 界面 如 图 14.2 所 示 ， 该 页 面 显示 了 游客 注册 时 需要 填写 的 昵称 、 密 码 和 E-mail 地 址 等 相 
关内 容 。 


es 
0 证 


wket TMT 
国王 


图 14.2 用 户 注册 信息 界面 
发 表 留 言 的 运行 效果 如 图 14.3 所 示 ， 该 页 面 用 于 注册 后 用 户 发 表 留 言 ， 包 括 留 言 主题 和 留言 内 容 。 
查询 页 面 运行 效果 如 图 14.4 所 示 ， 该 页 显示 执行 查询 后 所 查 到 的 留言 主题 、 留 言 内 容 ， 以 及 相关 回复 。 


2 
主题， 0 | 
Bs 
ee 本 CD 同人 辑 
Be a Tr 
多 让 由: WE ER 妆 CT9 开 菇 二 EN 和 ) 已 ET 
四 
ed ee LR 
一 和 和 
2 | 
ETT TY 
14.3 ”用 户 发 表 留 言 信息 界面 14.4 查询 界面 


14.3 ”数据 库 设 计 
殴 视频 讲解 : 光盘 \ITM\Video\ 第 14 章 \ 数 据 库 设计 .exe 
14.3.1 数据 库 概 要 说 明 
在 留言 本 中 ， 采 用 的 是 MySQL 数据 库 ， 用 来 存储 注册 用 户 的 基本 信息 、 留 言 信息 、 回 复 信息 和 管理 员 


信息 等 。 这 里 将 数据 库 命名 为 db_database14， 其 中 包含 的 数据 表 如 图 14.5 所 示 。 


国 服务 器 :lecalhesr 序数 据 库 : db_database14 
区 结构 HSQL 上 执 衬 局 寺 乔 冯 导 tb BImpor 区 要 作 翅 和 用 册 轴 革 


县 作 记录 着 全 。 关 型 至 理 到 小。 多才 

加 素面 X 1 Wisay gb2312_chinese ci 20KB 

EF 4 MGAM gb2312_cninsss_cl Zz7KB 144 亨 千 

伴生 重 XX 0 ItSAM gb2312_chinese_cl 29KB 926 字 季 

Ee 13 msAW gb2312_chinesa_ci 。 42kKB 杷 0 字 苇 
图 14.5 数据 库 结构 
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14.3.2 ”数据 库 概 念 设计 


根据 系统 分 析 和 功能 上 的 需要 ， 规 划 出 留言 本 中 数据 表 的 实体 E-R 图 ， 包 括 用 户 信息 实体 、 留 言 信息 
实体 、 回 复 信息 实体 和 管理 员 〈 版 主 ) 信息 实体 。 
用 户 信息 实体 包括 用 户 名 、 密 码 、 邮 箱 、 头 像 等 多 项 资料 ， 用 户 信息 实体 E-R 图 如 图 14.6 所 示 。 


留言 信息 实体 包括 留言 者 、 留 言 主题 、 留 言 内 容 、 留 言 时 间 等 。 留 言 信息 实体 E-R 图 如 图 14.7 所 示 。 


图 14.6 用 户 信息 实体 E-R 图 图 14.7 留言 信息 实体 E-R 图 
回复 信息 实体 包括 回复 者 、 回 复 主题 、 回 复 内 容 、 回 复 时 间 等 。 回 复 信 息 实体 E-R 图 如 图 14.8 所 示 。 
管理 员 〈 版 主 ) 信息 实体 包括 版 主 名 称 、 版 主 登 录 密码 、 版 主 邮箱 等 。 版 主 信息 实 体 E-R 图 如 图 14.9 


回复 者 回复 者 id 版 主 名 称 CH 
版 主 IP 
版 主 密码 ) \ 版 主 邮箱 
回复 主题 回复 内 容 回复 时 间 


图 14.8 回复 信息 实体 E-R 图 14.9 版 主 信息 实体 E-R 图 
14.3.3 ”数据 库 罗 辑 设 计 
根据 设计 好 的 E-R 图 在 数据 库 中 创建 数据 表 ， 下 面 给 出 重要 的 数据 表 结 构 。 


tb_user〈 用 户 信息 表 ) 
户 信息 表 用 于 存储 用 户 的 相关 信息 ， 用 户 信息 表 的 结构 如 表 14.1 所 示 。 


表 14.1 tb_user 表 的 结构 


用 户 密码 
用 户 真实 姓名 

用 户 E-mail 地 址 
用 户 QQ 号 码 
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续 表 


说 有明 
用 户 联系 电话 
用 户 注册 人 P 

用 户 联系 地 址 


Face varchar 用 户头 像 
Regtime | datetime | 用 户 注册 时 间 
Sex Varchar 用 户 性 别 


Usertype int 用 户 类 型 标记 


tb_ leaveword (留言 信息 表 ) 
留言 信息 表 存 储 了 用 户 留 言 的 相关 信息 ， 留 言 信息 表 的 结构 如 表 14.2 所 示 。 
表 14.2 tb_leaveword 表 的 结构 


字段 名 称 数 据 类 型 
Id int 
Userid int 
Createtime datetime 
Title varchar 
Content text 


tb_replyword (回复 信息 表 ) 
回复 留言 信息 表 用 于 存储 用 户 回 复 留言 的 信息 ， 回 复 留 言 信息 表 的 结构 如 表 14.3 所 示 。 


表 14.3 tb_replyword 表 的 结构 


字段 名 称 数据 类 型 说 了 明 
Id int 自动 编号 id 
Userid int 用 户 留言 id 
Createtimes datetime 回复 时 间 
Titles Varchar 回复 主题 
Contents text 回复 内 容 
Leave id int 回复 者 id 


tb_adm (管理 员 (版 主 ) 信息 表 ) 
管理 员 信息 表 用 于 存储 管理 员 的 信息 ， 管 理 员 信息 表 的 结构 如 表 14.4 所 示 。 


表 14.4 tb_adm 表 的 结构 


Userword 


版 主 名 称 
版 主 密码 
版 主 E-mail 地 址 
版 主 记 


password 


Email 
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14.4 公共 模块 设计 
名 4 视频 讲解 :光盘 \TM\Video\ 第 14 章 \ 公 共 模 块 设计 .exe 


14.4.1 数据 库 连接 文件 


在 进行 程序 开发 的 过 程 中 ， 有 很 多 的 地 方 涉及 到 数据 库 的 应 用 ， 在 应 用 数据 库 前 首先 要 与 数据 库 建立 
连接 ， 因 此 可 以 将 数据 库 的 连接 代码 作为 一 个 公共 文件 进行 存储 ， 在 需要 使 用 数据 库 连接 文件 的 地 方 直 接 
调用 该 文件 即 可 ， 无 需 重复 编写 相同 的 代码 ， 既 减少 代码 的 元 余 ， 也 便于 对 数据 库 连接 文件 进行 修改 。 在 
本 项 目 中 将 数据 库 的 连接 代码 存储 于 根 目 录 下 conn 文件 夹 的 conn.php 文件 中 。 

(代码 位 置 : 光盘 \TM\Instance\l4\connvconn.php) 

<?php 


$conn=mysql_connect("localhost","root","111"); // 连 接 数据 库 服务 器 
mysql_select_db("db_database14",$conn); // 连 接 指定 的 数据 库 

mysql_query("set names gb2312"); // 对 数据 库 中 的 编码 格式 进行 转化 ， 避 免 出 现 中 文 乱码 
Ss 


区 注意 由 于 本 机 的 MySQL 服务 器 的 用 户 名 为 root, 密码 为 111.。 读者 运行 本 程序 时 , 需要 在 conn.php 
文件 中 对 MySQL 服务 器 的 用 户 名 和 密码 进行 修改 ， 以 保证 程序 的 正常 运行 。 


成 功 创建 数据 库 conn.php 文件 后 ， 如 果 某 个 页 面 中 需要 进行 数据 库 的 操作 ， 在 页 面 中 直接 通过 include 
语句 包含 conn.php 文件 即 可 。 其 代码 如 下 : 
<?php 
include(“conn/conn.php”); 
?> 


14.4.2 ”将 文本 中 的 字符 转换 为 HTML 标识 符 


在 输出 数据 库 中 数据 的 过 程 中 ， 有 必要 将 数据 中 的 一 些 特殊 字符 转换 为 HTML 标识 符 ， 这 样 可 以 避免 
一 些 不 必要 的 麻烦 。 例 如 ， 在 输出 一 个 程序 的 执行 代码 的 过 程 中 ， 如 果 不 对 其 进行 转换 ， 那 么 输出 的 将 不 
是 程序 的 代码 ， 而 是 程序 的 执行 结果 。 这 里 将 文本 中 字符 的 转换 编写 到 一 个 自 定义 函数 unhtml0 中 ， 保 存 于 
function.php 中 ， 将 其 作为 一 个 公共 模块 来 使 用 ， 当 需要 使 用 时 直接 调用 function.php 文件 即 可 。 

function.php 文件 包含 两 个 自 定义 函 数 : unhtml0 和 msubstr0。unhtml0 函 数 用 于 将 数据 中 的 特殊 字符 转 
换 为 HTML 标识 符 ;，msubstr0 函 数 用 于 对 字符 串 进行 指定 长 度 的 截取 。 

《代码 位 置 ， 光盘 \TM\Instance\14\function.php) 


<?php 

function unhtml($content) { // 定 义 自 定义 函数 的 名 称 
oo $content=htmlspecialchars($content); /转换 文本 中 的 特殊 字符 
[2 S$content=str_replace("@","", $content); /| 替换 文本 中 的 换行 符 
目 return trim($content); /删除 文本 中 首尾 的 空格 


} // 定 义 一 个 用 于 截取 一 段 字符 串 的 函数 msubstr() 
function msubstr($str,$start,$len) { //$str 指 的 是 字符 串 ，$start 指 的 是 字符 串 的 起 始 位 置 ，$len 指 的 是 长 度 
$strien=$start+ $len; // 用 $strlen 存储 字符 串 的 总 长 度 〈 从 字符 串 的 起 始 位 置 到 字符 串 的 总 长 度 ) 
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S$tmpstr=""; /声明 一 个 变量 ， 并 赋值 为 空 
for($i=0;$i<$strlen;$i++){ // 通 过 for 循环 语句 读 取 字 符 串 
if(ord(substr($str,$i,1))>0xa0) { // 如 果 字符 串 首 个 字 节 的 ASCII 序数 值 大 于 0xa0， 则 表示 汉字 
S$tmpstr.=substr($str,$i,2); // 每 次 取出 两 位 字符 赋 给 变量 $tmpstr， 即 等 于 一 个 汉字 
Si++; /变量 自 加 1 
} else // 如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 $tmpstr 
@ Stmpstr.=substr($str, $i,1); 
} 
return $tmpstr; // 输 出 字符 串 


> 


me 
@str_replace(): 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替换 ， 参 数 &count 表示 替换 字 
符 串 执行 的 次 数 。 
@trim0: 删除 字符 囊 中 首尾 的 空白 或 者 其 他 字符 。 
@substr(): 从 指定 的 字符 串 中 按照 指定 的 位 置 截 取 一 定 长 度 的 字符 。 


14.4.3 JavaScript 脚本 


在 留言 本 中 ，JavaScript 脚本 一 般 用 于 表单 元 素 验证 ， 如 判断 text 文本 框 输入 是 否 为 空 ， 输 入 格式 是 否 
符合 标准 等 。 在 网 页 中 使 用 JavaScript 脚本 的 方式 主要 有 3 种 方式 。 
(1) 在 网 页 中 使 用 <script></scrip 人 标签 对 
<script></scrip 亿 标签 对 可 以 放 在 网 页 的 任意 位 置 ， 一 般 是 放 在 <head></head> 或 <body></body> 之 间 。 输 
出 系统 当前 时 间 的 JavaScript 脚本 代码 如 下 : 
<script language=JavaScript> 
today=new Date(); 
function initArray(X{ 
this.length=initArray.arguments.length 
for(var i=0;i<this.length;i++) 
this[i+1]=initArray.arguments[i] } 


var d=new initArray( 


"星期 一 ", 
"星期 二 "， 
"星期 三 ", 
"星期 四 "， 
"星期 五 "， 
"星期 六 "); 
document.write( 
"<font color=#000000 style='font-size:9pt;font-family: 宋体 > "， 
today.getYear()," 年 "， 
today.getMonth()+1," 月 "， 
today.getDate()," 日 "， 
"gnbsp;&nbsp;", 
Wd .getDay()+1], 
"</font>" ); 
</script> 


(2) 在 单独 文件 中 使 用 
如 果 JavaScript 脚本 比较 多 ,而 且 位 置 分 散 不 易 管理 ， 可 以 统一 放 到 一 个 扩展 名 为 js 的 文件 中 ,使 该 文 
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件 成 为 JavaScript 脚本 文件 。 在 脚本 文件 中 ， 不 需要 使 用 <scripf></scripf> 标 签 对 ， 直 接 写 脚本 代码 即 可 。 当 
页 面 需要 使 用 到 里 面 的 JavaScript 脚本 时 ， 可 以 这 样 引用 ， 代 码 如 下 : 
<SCRIPT src="calendar.js"></SCRIPT> 
<SCRIPT Inguage="Javascript"> 
var xhnetCalendar=new calendar("xhnetCalendar","xhnetCal"); 
</SCRIPT> 
<SCRIPT> xhnetCalendar.channel=forum';</SCRIPT> 
<DIV class=calendar id=abc> 
<SCRIPT> 
document.write(xhnetCalendar.generate(null)); 
xhnetCalendar.focusDate(); 
</SCRIPT> 


人 让 总 本 留言 本 主页 上 的 万 年 历 就 是 采用 这 种 方式 引用 的 。 万 年 历 的 主要 代码 请 参见 光盘 的 
calendarjs 文件。 

(3) 在 表单 元 素 或 标签 中 使 用 

这 是 最 直接 的 使 用 方式 ， 如 果 是 少量 的 脚本 则 可 以 这 样 使 用 。 例 如 ， 在 超 链接 标签 <a> 中 想 要 使 用 


JavaScript 脚本 ， 代 码 格式 如 下 : 
<a href="reply.php?t_id=<?php echo $_GET[1Lid]; ?>&loor=<?php echo $i;?>"> 回 复 </a> 


在 留言 本 中 ， 这 3 种 方法 都 有 使 用 到 ， 在 后 面 涉及 到 具体 应 用 时 再 进行 说 明 。 


14.5 首页 模块 设计 
铭 i 视频 讲解 :光盘 \TM\Video\ 第 14 章 \ 首 页 模块 设计 .exe 


14.5.1 首页 设计 概述 


本 系统 首页 页 面 设计 得 比较 简洁 明了 ， 7 

主要 包括 以 下 4 部 分 内 容 。 衣 保 个 m。 a < 
首部 导航 栏 : 包括 首页 链接 、 用 户 TT 
注册 、 发 表 留言 、 查 看 留言 、 查 询 | = 


留言 、 版 主管 理 和 注销 登录 模块 。 
回 左 侧 显 示 区 : 包括 用 户 登录 、 万 年 
历 和 最 新 留言 模块 。 游 客 通过 该 区 
域 登录 留言 本 ， 以 及 了 解 最 新 的 留 


i | 
EET 


言 主题 。 pe 有 
回 “ 主 显示 区 : 为 留言 本 的 最 新 留言 。 8 ee 
游客 通过 该 区 域 可 以 查看 最 新 的 贸 
言 主题 与 留言 内 容 。 
回 “尾部 显示 区 : 为 留言 本 的 版 权 声明 。 
留言 本 首页 效果 图 如 图 14.10 所 示 。 图 14.10 主 界面 


GB 


14.5.2 switch 和 include 语句 


留言 本 首页 设计 主要 应 用 switch 语句 和 include 包含 语句 ， 其 实现 的 原理 是 : 应 用 switch 语句 ， 根 据 超 
链接 中 传递 的 变量 值 进行 判断 ， 根 据 不 同 的 变量 值 应 用 include 调用 不 同 的 子 文件 。 
为 了 更 好 地 理解 这 个 技术 ， 先 来 了 解 一 下 switch 语句 。 该 语句 的 格式 如 下 : 
switch(exprjf /expr 条 件 为 变量 名 称 
case expr1: licase 后 的 expr1 为 变量 的 值 
statement1; // 冒 号 (:) 后 的 是 符合 该 条 件 时 要 执行 的 部 分 
break; /应 用 break 来 跳 离 循环 体 
Case expr2: 
statement2; 
break; 
default: 
statementN; 
break; 
出 
参数 expr 是 表达 式 的 值 ， 即 switch 语句 的 条 件 变 量 的 名 称 ; 参数 exprl 放置 于 case 语句 后 ， 是 要 与 条 
件 变量 expr 进行 匹配 的 值 中 的 一 个 ;statementl 是 参数 exprl 的 值 与 条 件 变 量 expr 的 值 相 匹配 时 执行 的 代码 ; 
break 语句 实现 终止 语句 的 执行 ， 即 当 语 句 在 执行 过 程 中 ， 遇 到 break 就 停止 执行 ， 跳 出 循环 体 ，default 是 
case 的 一 个 特例 ， 匹 配 了 任何 其 他 case 都 不 匹配 的 情况 ， 并 且 是 最 后 一 条 case 语句 。 
通过 switch 和 include 语句 来 实现 首页 的 设计 是 一 个 很 好 的 方法 ， 不 但 实现 过 程 简单 ， 而 且 操 作 非 常 
活 。 其 关键 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\14\index.php) 
<?php 
Switch($id){ // 根 据 变量 提交 的 不 同 值 
case "首页 ": // 判 断 与 变量 提交 的 值 是 否 相同 
include "main.php"; // 如 果 值 相 同 ， 则 调用 指定 文件 
break:; // 并 且 跳 出 本 次 循环 
case "用 户 注册 ": 
include "reg.php"; 
break; 
case "发 表 留言 … 
include "leaveword.php"; 
break; 
case "查看 留言 ": 
include "lookleaveword.php"; 
break; 
case "查询 留言 ": 
include "searchword.php"; 
break; 
case "版 主管 理 ": 
include "login.php"; 
break; 
case "注销 登录 ": 
include "logout.php"; 
break; 
case "编辑 留言 
include "editleaveword.php"; 
break; 
case "回复 编辑 留言 "- 


灵 
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include "edlitreplyword.php"; 

break; 

case "详细 信息 
include "lookxx.php"; 

break; 

default: // 判 断 当 该 值 等 于 空 时 ， 调 用 该 文件 
include "main.php"; 

break; 

} 


?> 


14.5.3 ”首页 实现 过 程 


在 一 个 网 站 中 ， 首 页 被 访问 的 次 数 是 比较 多 的 。 为 了 加 快 页 面 的 运行 速度 、 提 高 访问 量 ， 本 项 目 首页 
使 用 include 语句 包含 主要 功能 模块 。 其 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\14\index.php) 


<?php 

© include_once("top.php"); 

?> 

<table width="779" height="23" border="0" align="center ”cellpadding="0" cellspacing="0"> 
<tr> 


<td width="292" background="images/dh_back.gif"><div align="right"> 今 天 是 : &nbsp; 

<script language=JavaScript> 

today=new Date(); 

function initArray(){ 

this.length=initArray.arguments.length 

for(var i=0;i<this.length;i++) 

this[i+1]=initArray.arguments[i] } 

var d=new initArray( 


"星期 六 "); 
document.write( 
"<font color=#000000 style='font-size:9ptfont-family: 宋体 > "， 
today.getYear()," 年 ", 
today.getMonth()+1," 月 "， 
today.getDate()," 日 "， 
"&nNnbsp;&nbsp;", 
dftoday.getDay()+1], 
"</font>" ); 
</script></div></td> 
© <td width="200" valign="top"><?php include_once("left.php");?></td> 
</td> 
© < 上 -留言 信息 --> 
</tr> 
</table> 
<?php 
9 include_once("bottom.php"); 


?> 
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~ 
SA 明 @ 应 用 include 语句 包含 top php 文件 ， 该 文件 用 于 显示 网 站 导航 、 留 言 本 名 称 及 当前 登录 的 用 


户 名 称 。 


四 应 用 include 语句 包含 leftphp 文件 ， 该 文件 用 于 显示 用 户 登录 、 万 年 历 及 最 新 留言 信息 。 
四 在 首页 (index.php ) 中 ， 应 用 表格 布局 的 方式 显示 留言 内 容 。 
四 应 用 include 语句 包含 bottom.php 文件 ， 该 文件 用 于 显示 版 权 信息 。 


14.6 ”用户 注册 模块 设计 


区 Nd 视频 讲解 : 光盘 \TM\Video\ 第 14 章 \ 用 户 注册 模块 设计 .exe 


14.6.1 ”用户 注册 模块 概述 

留言 本 为 了 更 好 地 与 用 户 进行 交流 和 沟通 ， 创 建 了 一 
个 用 户 注 册 模块 ， 通 过 用 户 注册 模块 ， 可 以 有 效 地 对 用 户 
信息 进行 采集 ， 并 将 合法 的 用 户 信息 保存 到 指定 的 数据 表 
中 ， 实 现 与 用 户 的 长 期 沟通 和 交流 。 用 户 注册 模块 的 运行 
结果 如 图 14.11 所 示 。 


14.6.2 ” JavaScript 脚本 验证 表单 元 素 


在 用 户 注册 模块 中 ， 必 不 可 少 的 就 是 要 对 用 户 输入 的 
信息 进行 判断 ， 首 先 判断 用 户 填写 的 注册 信息 中 哪些 是 必 
须 填写 的 ， 哪 些 可 以 不 填写 ， 然 后 进一步 判断 输入 的 信息 
是 否 合理 合法 ， 如 判断 输入 的 邮箱 的 格式 是 否 正确 、 电 话 
号 码 是 否 有 效 等 。 对 表单 中 提交 的 数据 进行 判断 最 常用 的 


ME 
二 用 声 明 


i 
2) 挫 和 式 表土 队 事 坟 ， 芭 这 寺 ， 状 对 守 会友 的 ， 


FRR。 这 
志 到 eeeeeee 
Rr: eeeeeeee 
Mt 字 纺 
人 性 咕 康 


14.11 用 户 注册 模块 的 运行 结果 


办 法 就 是 使 用 JavaScript 脚本 ， 也 可 以 使 用 正则 表达 式 。 下 面 讲解 在 本 模块 中 是 如 何 通过 JavaScript 脚本 实 


现 表 单 提交 数据 的 验证 。 


操作 原理 是 : 在 form 表单 中 调用 onSubmit 事件 ， 通 过 该 事件 调用 指定 的 JavaScript 脚本 ， 执 行 自 定义 
函数 chkinputO， 实 现 对 表单 中 提交 数据 的 验证 。 在 JavaScript 脚本 中 ， 实 现 对 表单 中 提交 数据 的 判断 ， 判 
断 输 入 的 内 容 是 否 为 空 、 内 容 格式 是 否 正 确 ， 如 果 正 确 则 继续 执行 ， 否 则 将 弹出 提示 对 话 框 ， 并 将 鼠标 的 


焦点 指定 到 出 错 的 位 置 。 具 体 的 JavaScript 脚本 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\14\reg.php) 
<script language="javascript"> 


function chkinput(form){ /定义 一 个 函数 
if(form.usernc.value=="™" /判断 usernc 文本 框 中 的 值 是 否 为 空 
alert(" 请 输入 用 户 昵称 !"); // 如 果 为 空 则 输出 “请 输入 用 户 昵称 ” 
form.usernc.focus(); 1/ 返回 到 tel 文本 框 
return(false); 
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ifform.userpwd.value=="){ 
alert(" 请 输入 注册 密码 !"); 
form.userpwd .focus(); 
return(false); 
3 
if(form.userpwd1.value=="™"){ 
alert(" 请 输入 重复 密码 !"); 
form.userpwd1.focus(); 
return(false); 
3 
if(form.userpwd.value!=form.userpwd1.value}{ 


alert(" 密 码 与 确认 密码 不 同 !"); 
form.userpwd.focus(); 
return(false); 


var i=form.email.value.indexOf("@"); 
var j=form.email.value.indexOf("."); 
if((i<O)Ij>0)l1G<0)X 
alert(" 请 输入 正确 的 E-mail 地 址 ""); 
form.email.select(); 
return(false); 


小 
return(true); /提交 表单 
} 
</script> 
上 述 代 码 中 ， 只 是 列举 了 JavaScript 中 的 部 分 内 容 ， 并 且 在 对 电话 号 码 进行 判断 时 ， 只 是 判断 其 是 是 否 


八 下 
为 数字 ， 没 有 进一步 判断 电话 号 码 的 格式 是 否 正 确 。 如 果 想 要 更 加 准确 地 判断 电话 号 码 的 格式 是 否 正确 可 
以 采用 下 面 的 方法 。 


通过 正则 表达 式 的 preg_match0 函 数 ， 在 表单 提交 处 理 页 中 对 电话 号 码 进行 判断 。preg_match0 函 数 的 
语法 格式 如 下 : 

int preg_match ( string pattern, string subject [, array matches [, int flags]] ) 

preg_match() 函 数 的 参数 说 明 如 表 14.5 所 示 。 

表 14.5 preg_match() 函 数 的 参数 说 明 
参数 说 有明 

patterm 必 选 参数 。 需 要 匹配 的 正则 表达 式 
subject | 必 选 参数 。 输 入 的 字符 串 


可 选 参数 。 输 出 搜索 结果 的 数组 ， 如 Sout[0] 将 包含 与 整个 模式 匹配 的 结果 ，S$out[]] 将 包含 与 第 一 个 捕获 
的 括号 中 的 子 模式 所 匹配 的 结果 ， 依 次 类 推 


matches 


可 选 参数 。 若 设 为 PREG_ OFFSET_ CAPTURE， 对 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 串 偏 移 
2 量 ， 本 标记 自 PHP 4.3.0 起 可 用 


通过 preg_matchO 函 数 判 断 电 话 号 码 的 格式 是 否 正确 的 方法 如 下 。 

首先 定义 一 个 用 于 判断 电话 号 码 格式 的 正则 表达 式 。 代 码 如 下 : 

人 df{3}-)(df8)SIA(d(4-)Cdf7DSIACd(4-)Cd(8DS/ 

正则 表达 式 的 功能 分 析 如 下 : 使 用 “^” 和 “$” 对 字符 串 进行 边界 的 限制 ， 对 区 号 从 字符 串 的 开始 进行 
匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 ， 将 括号 “QO” 中 的 内 容 作为 一 个 原子 使 用 ; 使 用 “\d” 来 


@ 


第 14 章 ”综合 实例 (二 ) 一 一 留言 本 


一 个 数字 , 区 号 为 3 或 4 个 数字 ,其 他 数字 为 7 或 8 个 ; 使 用 “f” 来 对 前 字符 进行 重复 匹配 ; 使 用 “|” 
4 匹配 的 模式 进行 选择 ， 分 成 3 个 模式 。 

然后 将 该 正则 表达 式 应 用 到 preg_match0 函 数 中 ， 对 表单 提交 的 电话 号 码 进行 判断 ， 如 果 正 确 则 继续 执 
行 ， 否 则 弹出 提示 信息 ， 并 返回 到 表单 提交 页 。 代 码 如 下 : 


妆 四 


<?php 
S$tel="0431-84978981"; // 定 义 一 个 电话 号 码 的 变量 
if(preg_match("/(\d{3}-)(\d{8})$I^(\d{4}-)(\d{7})$I^(d{4}-)(\d{8})$I^(d{11})$/", Stel,$counts)\{ 
echo "您 输入 的 电话 号 码 格式 正确 !"; /输出 字符 串 
Jelse{ 


echo "<script>alert(' 您 输入 的 电话 号 码 的 格式 不 正确 !);history.back()</script>"; 
} 


?> 


14.6.3 ”用 户 注 册 模 块 实现 过 程 


注册 模块 的 实现 过 程 非常 简单 ， 首 先 阅 读 注 册 声明 ， 然 后 填写 用 户 注册 的 用 户 名 和 密码 ， 提 交 后 由 系 
统 判断 输入 的 用 户 名 是 否 被 占用 ， 如 果 未 被 占用 则 可 以 继续 注册 ， 将 数据 提交 到 表单 处 理 页 进行 处 理 ， 最 
后 将 用 户 注册 的 信息 保存 到 指定 的 数据 表 中 。 

用 户 注册 模块 由 两 个 文件 组 成 : reg.php 文件 用 于 填写 详细 的 注册 信息 ;savereg.php 文件 用 于 对 表单 中 
提交 的 数据 进行 处 理 ， 将 数据 保存 到 指定 的 数据 表 中 。 
在 savereg.php 文件 中 ， 首 先 连 接 数据 库 ， 然 后 获取 到 表单 中 提交 的 数据 ， 并 且 判 断 提交 的 用 户 名 是 否 
被 占用 ， 最 后 将 提交 的 数据 进行 处 理 ， 并 将 数据 保存 到 指定 的 数据 表 中 。 程 序 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\14\savereg.php) 


<?php 
session_start(); /启用 session 支持 
if(isset($_SESSION['userword"])X{ // 获 取 登 录 的 版 主 名 称 


echo "<script>alert(' 在 同一 台 机 器 上 ， 不 允许 同时 使 用 用 户 名 和 管理 员 进 行 登录 ! '); 
window.location.href='index.php';</script>";， ”// 给 出 用 户 与 版 主 不 能 同时 登录 的 信息 


Jelse{ 
include_once("conn/conn.php"); // 包 含 数 据 库 文件 
$usernc=$_POST["userne"]; /调用 注册 时 提交 的 用 户 名 称 
$sql=mysql_query("select usernc from tb_user where usernc=".$usernc."",$conn); 
/用户 家 中 已 注 出 用 上 的 宫 各 生生 与 当前 提交 的 用 户 名 相同 
$info=mysql_fetch_array($sql); // 输 出 查询 的 数据 
if$info 
echo "<script>alert(' 对 不 起 ， 你 的 昵称 已 经 被 占用 ! '");history.back();</script>"; /相同 的 给 出 提示 
exit; 


} 
S$ip=$_SERVER["REMOTE_ADDR"]; 
if(mysql_query("insert into tb_user(usernc,userpwd,truename,email,qq,tel,ip,address ,face,regtime,sex,usertype) 
values(".$usernc.",".md5(tim($_POST["userpwd"])).",".$_POST["truename"].",".$_POST["email"].",".$_POS 
Tr'qq"].",".$_POST['tel"].",".$ip.",”".$_POST["address"].",".$_POST['"face"].",".date("Y-m-d 
Hii:s").",".$_POST["sex"].",'0")",$conn)) 


{ 
$_SESSION["unc"]=$usernc; 
echo "<script>alert(' 注 册 成 功 ! ');window.location.href='index.php'</script>"; 
jelse{ 
echo "<script>alert(' 注册 失败 ! ');history.back();</script>"; 
} 
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14.7 添加 留言 模块 设计 


14.7.1 添加 留言 模块 概述 


当 用 户 登 录 后 ， 单 击 首页 上 的 “发 表 留 言 ” 超 链接 ， 将 加 本 
载 发 表 留言 表单 。 用 户 填写 完 留言 标题 、 留 言 内 容 后 , 单 击 “ 发 sa 
表 ” 按 钮 ， 跳 转 到 留言 处 理 页 进行 处 理 。 用 户 添加 的 留言 将 会 
保存 到 MySQL 数据 库 中 。 添 加 留言 的 界面 如 图 14.12 所 示 。 


14.7.2 mysql_query() 函 数 执行 SQL 语句 | 


将 用 户 发 表 的 留言 保存 到 MySQL 数据 库 是 通过 SQL 语言 图 14.12 用 户 添加 留言 的 运行 结果 
的 insert 语句 实现 的 。PHP 通过 mysql query0 函 数 向 MySQL 
数据 库 中 发 送 SQL 命令 ， 所 以 mysql_query0 函 数 在 数据 库 编程 过 程 中 具有 极其 重要 的 作用 ， 下 面 对 该 函数 
进行 介绍 。 
mysql_query0 函 数 用 于 向 与 指定 的 连接 标识 符 所 关联 的 数据 库 服务 器 中 发 送 一 条 查询 语句 。 其 语法 如 下 : 
resource mysql_query (string query [, resource link_identifier]) 
query: 必 选 参数 。 用 于 指定 向 数据 库 中 发 送 的 查询 语句 。 
link_identifier: 可 选 参数 。mysql_connect0 函 数 成 功 连接 数据 库 后 所 返回 的 连接 标识 ， 如 果 省 略 了 
该 参数 ， 则 使 用 最 近 一 次 与 数据 库 建 立 连接 后 所 返回 的 连接 标识 。 


14.7.3 ”添加 留言 模块 实现 过 程 


添加 留言 模块 ， 首 先 应 建立 用 于 留言 的 表单 ， 该 过 程 实现 相对 简单 ， 这 里 不 再 袭 述 。 

然后 ， 获 取 表单 中 提交 的 数据 ， 对 用 户 输入 的 留言 信息 进行 敏感 词 过 滤 ， 将 用 户 留言 内 容 保 存 到 数据 
库 中 。 其 具体 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\l4\saveleaveword.php) 


<?php 

session_start(); /启用 session 支持 

include_once("conn/conn.php"); // 包 含 数 据 库 连 接 文 件 

include_once("function.php"); // 包 含 系统 功能 文件 

if(isset($_SESSION["unc"])X{ // 对 登录 用 户 进行 判断 
$sql=mysql_query("select id from tb_user where usernc=".$_SESSION["unc"]."”); 

/查询 当前 用 户 数据 表 中 的 信息 

$info=mysql_fetch_array($sql); /从 结果 集中 获取 信息 
$userid=$inforid]; /获取 用 户 的 id 


} 

iflisset($_SESSION[userword])X{ 
$sql=mysql_query("select id from tb_adm where userword=".$_SESSION["userword"]."™"); 
Sinfo=mysql_fetch_array($sql); 
$userid=$info[id]; 


} 
if(isset($_POST[content])X{ 


局 
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@ if (is_file("filterwords.txt")X{ /判断 给 定 文件 名 是 否 为 一 个 正常 的 文件 
Sfilter_word = file("filterwords .txt"); // 把 整个 文件 读 入 一 个 数组 中 
$content=$_POSTTcontent]; 
$title=$_POSTTtitle]; 
for($i=0;$i<count($fiter_word):$i++)f /应 用 for 循环 语句 对 敏感 词 进行 判断 

@ 说 preg_match(v"trim($filter_word[$il).wi",$content){ ”// 应 用 正则 表达 式 ， 判 断 传递 的 留言 信 

息 中 是 否 含有 敏感 词 
echo "<script>alert( 留言 信 息 中 包含 敏感 词 ! ');history.back(-1);</script>"; 
exit; 

5 b 
3 Screatetime=date("Y-m-d H:i:s"); // 获 取 留 言 时 间 


if(mysql_query("insert into tb_leaveword(userid,createtime,title,content)values($userid',$createtime',$title 
‘$content)")X 
echo "<script>alert(' 留言 发 表 成 功 ! ');window.location.href='index.php?id=".urlencode(' 查 看 留言 
").";</script>"; 
证 在 页 面 中 提交 的 留言 信息 ， 写 入 到 数据 库 留 言 表 中 ， 提 示 留 言 成 功 并 跳 转 到 查看 留言 


echo "<script>alert(' 留 言 发 表 失败 ! ');history.back();</script>"; 
} 


?> 


/ 
说 日 
Be: 说 明 @is file0 函 数 : 判断 该 函数 参数 所 表示 的 文件 是 否 存在 并 且 为 一 个 合法 文件 。 
@ preg_matchO 函 数 : preg_match0 函 数 中 的 “/i”， 是 指 在 进行 敏感 词汇 比较 时 区 分 字母 大 小 写 。 
@date0 函 数 : 格式 化 一 个 本 地 时 间 / 日 期 。 


14.8 查看 留言 模块 设计 
多 i 视频 讲解 : 光盘 \TM\Video\ 第 14 章 \ 查 看 留言 模块 设计 .exe 


14.8.1 查看 留言 模块 概述 


用 户 登 录 后 ， 单 击 首页 上 的 “查看 留言 ”链接 ， 将 进入 富 守 m 
浏览 页 面 。 在 浏览 页 面 中 ， 用 户 除了 可 以 看 到 留言 信息 外 ， 网 
还 可 以 看 到 对 该 留言 的 回复 。 如 果 留 言 或 者 回复 是 当前 用 户 铁 1 天 生计 9] 和 于 忆 疹 ， 请 大 人 贡 隐 时 各 
所 发 表 ， 那 么 还 会 看 到 修改 的 超 链接 。 如 果 回 复 是 当前 用 户 。 Se vs 
所 发 表 的 , 那么 除 显示 上 述 超 链接 外 , 还 显示 删除 的 超 链接 。 
查看 留言 页 面 的 运行 结果 如 图 14.13 所 示 。 RE 

伟 入 下 37 地址 和 a 

14.8.2 ”验证 数据 类 型 与 取 整 es 0 


第 一 疮 、 尖 一 当 各 本 过 to 十 实 齐 和 :on 外 技 
er 全 友和 二 
吝 “村 ， 天 家 中 也 短 可 术 ， 扩 在 十 几 


用 户 发 表 完 留言 后 , 可 以 通过 查看 留言 模块 查看 用 户 的 。 -aaamuas al 
所 有 留言 内 容 。 由 于 用 户 的 留言 数 日 较 多 ， 如 果 在 同一 页 面 。 "= sse 和 人 EE 


图 14.13 查看 留言 界面 
-四 
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中 显示 所 有 留言 信息 则 会 给 用 户 浏览 带 来 很 大 的 不 便 ， 所 以 通过 分 页 的 方式 显示 用 户 留 言 内 容 是 不 错 的 选 
择 。 在 实现 用 户 留言 内 容 分 页 显示 时 ， 主 要 应 用 is_numericO 函 数 判断 用 户 通过 GET 方法 提交 的 数据 是 否 为 
数值 型 ， 并 通过 ceil0 函 数 对 页 码 数据 进行 向 上 取 整 。 

1. is_numeric() 函 数 

如 果 该 函数 的 参数 为 数字 或 数字 字符 串 则 返回 True， 和 否则 返回 False。 语 法 如 下 : 

bool is_numeric ( mixed var ) 

参数 var 为 要 进行 判断 的 数据 。 

2. ceil() 函 数 

ceil0 函 数 用 于 对 浮 点 数 进行 向 上 取 整 。 语 法 如 下 : 

float ceil ( float value ) 

参数 value 为 要 进行 向 上 取 整 的 数据 。 


14.8.3 ”查看 留言 模块 实现 过 程 


在 查看 留言 模块 中 ， 以 分 页 的 形式 循环 输出 留言 信息 ， 并 且 根据 对 当前 用 户 权限 设置 一 些 具 体 的 操作 
链接 是 否 输出 。 下 面 是 显示 用 户 留 言 信息 的 关键 代码 。 
(代码 位 置 ， 光盘 \TM\Instance\14\lookleaveword.php) 
<?php 
$sql=mysql_query("select count(*) as total from tb_leaveword ",$conn); 
$infos=mysql_fetch_array($sql); 


Stotal=$infos['total]; // 获 取 总 留言 条 数 
if($total==0X{ // 如 果 总 留言 条 数 为 0， 则 给 出 提示 
echo "<div align=center> 对 不 起 ， 暂 无 留言 ! </div>"; 
Jelse{ 
if(!isset($_GET["page"]) || !is_numeric($_GET["page"])X{ 
$page=1; // 判 断 查 询 字符 串 page 的 值 是 否 为 空 ， 如 果 为 空 则 默认 显示 第 1 页 
}else{ 
$page=intval($_GET["page"]); 
} 
$pagesize=3; // 规 定 每 页 显示 3 条 留言 
if($total%$pagesize==0X{ // 获 取 总 页 数 
$pagecount=intval($total/$pagesize);  // 如 果 获 取 的 总 页 数 是 整数 ， 则 返回 整数 值 
Jelse{ 


S$pagecount=ceil($total/$pagesize); // 获 取 的 总 页 数 不 是 整数 时 ， 进 行 向 上 取 整 


$sql=mysql_query("select * from tb_leaveword order by createtime desc limit ".($page-1)*$pagesize.",$pagesize 
",$conn); 
while($info=mysql_fetch_array($sql)X{ // 通 过 while 循环 显示 所 有 留言 
$sql1=mysqlL_query("select usernc,face,ip,email,qq from tb_user where id=".$info["userid"]."",$conn); 
// 通 过 传递 的 id 值 查询 当前 留言 者 的 个 人 信息 
$info1=mysql_fetch_array($sql1); 
7 
a // 显 示 用 户 信息 和 用 户 留言 信息 
<?php 
$adms=mysql_query("select id from tb_adm where userword=".$adm."",$conn); 
// 从 管理 员 数 据 表 中 查询 登录 的 版 主 id 
S$re=mysql_fetch_array($adms); 


if($rerid]==Sinfo['userid] && $adm!="X{// 判 断 当 前 登录 的 管理 员 是 不 是 该 条 留言 的 发 表 者 ， 如 果 是 则 显示 编辑 按钮 
?> 


ee // 显 示 编辑 按钮 
@ 
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<?php 
}elseif($user== $info1[usernc] and $user!="™"X{ 1/ 如果 当前 用 户 是 该 留言 的 发 表 者 ， 则 显示 编辑 按钮 
?> 
ee // 显 示 编辑 按钮 
<?php 
}else{ 
?> 
// 编 辑 按钮 为 灰色 ， 即 不 显示 
<?php 
} 
> 
<?php 
if($adm!="™"X // 管 理 员 登 录 后 ， 可 以 对 帖子 进行 回复 
?> 
es // 显 示 回 复 此 帖 
<?php 
}else if($user!= $info1[usernc] and $user!="X{ // 当 前 用 户 不 是 该 帖子 的 发 表 者 时 , 可 以 对 帖子 进行 回复 
?> 
ee // 显 示 回 复 此 帖 
<?php 
jelsef 
?> 
<?php 
} 
?> 


上 述 代 码 中 ， 实 现 留 言 信息 的 分 页 显示 ， 是 通过 MySQL 数据 库 的 扩展 关键 字 limit 来 实现 的 ， 该 关键 
字 后 跟 两 个 参数 。 其 中 ， 第 一 个 参数 用 于 指定 要 显示 记录 的 起 始 位 置 ， 而 第 二 个 参数 用 于 指定 所 要 显示 的 
记录 个 数 。 
在 显示 用 户 留 言 时 ， 根 据 超 链接 传递 的 page 值 来 决定 所 要 显示 的 记录 范围 。 用 于 实现 向 页 面 中 传 入 查 
询 字符 串 的 关键 代码 如 下 : 
<table width="550" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
<td width="351"><div align="left"> 共 有 留言 &nbsp;<?php echo S$total;?>&nbsp; 条 &nbsp; 每 页 显示 
&nbsp;<?php echo $pagesize;?>&nbsp; 条 &nbsp; 第 &nbsp;<?php echo $page;?>&nbsp; 页 / 共 &nbsp;<?php echo 
$pagecount;?>&nbsp; 页 </div></td> 


<td width="199"><div align="right"> 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=1&id=<?php echo urlencode($id);?>" 
class="a1"> 首 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 
if($page>1) /判断 当前 页 是 否 大 于 第 一 页 ， 如 果 是 则 当 用 户 单 击 “ 上 一 页 ” 超 
链接 时 ， 使 变量 $page 的 值 减 1， 从 而 实现 向 前 翻 页 的 功能 
echo $page-1; 
else 
echo 1; 
?>&id=<?php echo urlencode($id);?>" class="a1"> 上 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 
if($page<$pagecount)  // 判 断 当前 页 码 是 否 小 于 总 的 页 数 , 如 果 是 则 当 用 户 单 击 “ 下 一 页 ” 
超 链 接 时 ， 使 变量 $page 的 值 加 1， 从 而 实现 向 前 翻 页 的 功能 
echo $page+1; 
else 
echo $pagecount; 
?>&id=<?php echo urlencode($id);?>" class="a1"> 下 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php echo $pagecount;?>&id=<?php 
echo urlencode($id);?>" class="a1"> 尾 页 </a></div></td> 
</tr> 
四 
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</table> 


由 


14.9 编辑 留言 模块 设计 


14.9.1 编辑 留言 模块 概述 

当 用 户 浏览 自己 发 表 的 留言 或 回复 其 他 用 户 留言 时 , 会 
显示 编辑 功能 。 用 户 可 以 通过 这 个 编辑 链接 ， 对 已 发 表 的 内 
容 进行 修改 。 编 辑 留言 页 面 的 运行 结果 如 图 14.14 所 示 。 


14.9.2” ”JavaScript 脚本 控制 弹出 对 话 框 ee [em 


图 14.14 编辑 留言 页 面 的 运行 结果 


在 本 项 目 中 , 采用 弹出 窗口 显示 预 编辑 的 留言 内 容 。 通 
过 JavaScript 实现 弹出 窗口 ， 主 要 是 通过 调用 window 对 象 的 open0 方 法 实现 ， 并 将 实现 调用 的 代码 封装 到 
-个 单独 的 自 定义 方法 中 ， 最 后 通过 HTML 元 素 的 onclick0 事 件 对 该 方法 进行 调用 。 
open0 方 法 的 语法 如 下 : 
open("url","name","features") 
该 方法 参数 说 明 如 表 14.6 所 示 ， 参 数 features 的 常规 特征 如 表 14.7 所 示 。 


表 14.6 open() 方 法 参数 说 明 


参数 取 值 说 明 
a 可 选 参数 。 用 于 指定 弹出 新 窗口 的 URL 地址， 如果 省 略 该 参数 或 它 的 值 是 一 个 空 字符 串 ， 
那么 新 窗口 就 不 显示 任何 文档 
a 新 窗口 的 名 字 , 如 果 这 个 名 字 是 一 个 已 经 存在 的 窗口 , 那么 open0 方 法 就 不 再 创建 一 个 新 窗 
口 ， 而 只 是 返回 这 个 窗口 的 引用 ， 在 这 种 情况 下 参数 features 将 被 忽略 
an 该 窗口 主要 是 显示 窗口 的 一 些 特征 , 该 参数 如 果 不 设 定 则 显示 这 个 窗口 的 所 有 特征 .features 
的 常规 特征 请 参见 表 14.7 
表 14.7 参数 features 的 常规 特征 
名 称 用 法 
channelmode 指定 窗口 是 否 应 该 以 频道 的 方式 显示 ， 取 值 有 yes/no 或 1/0 
fullscreen 指定 窗口 是 否 全 屏 显示 ， 取 值 有 yes/no 或 1/0 
menubar 指定 窗口 是 否 有 菜单 栏 ， 取 值 有 yes/no 或 1/0 
scrollbars 指定 窗口 是 否 有 水 平 及 竖 直 滚动 条 ， 取 值 有 yes/no 或 1/0 
status 指定 窗口 是 否 有 状态 栏 ， 取 值 有 yes/no 或 1/0 
toolbar 指定 窗口 是 否 有 工具 栏 ， 取 值 有 yes/no 或 1/0 
location 指定 窗口 是 否 有 地 址 栏 ， 取 值 有 yesmo 或 1/0 
directions 指定 新 建 窗口 是 否 有 标准 目录 按 扭 ， 取 值 有 yes/no 或 1/0 
Tesizable 指定 新 窗口 大 小 是 否 可 以 调整 ， 取 值 有 yesino 或 1/0 
top 以 像素 为 单位 指定 新 窗口 上 沿 与 屏幕 上 沿 的 距离 


left 以 像素 为 单位 指定 新 窗口 左 沿 与 屏幕 左 沿 的 距离 
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width 以 像素 为 单位 指定 新 窗口 的 宽度 
height 以 像素 为 单位 指定 新 窗口 的 高 度 


14.9.3 ”编辑 留言 模块 实现 过 程 


为 了 保证 用 户 的 留言 内 容 不 被 他 人 私自 更 改 ， 在 具体 实现 该 模块 功能 时 ， 采 用 每 个 用 户 只 能 对 自己 的 
留言 内 容 进 行 更 改 的 方式 。 具 体 实现 时 ， 将 标识 用 户 身份 的 session 变量 的 值 与 数据 库 中 该 条 留言 对 应 的 留 
言 者 进行 比较 ， 如 果 二 者 相同 ， 则 说 明 该 条 留言 为 当前 登录 用 户 所 发 表 的 ， 在 该 条 留言 后 显示 “编辑 ” 按 
钮 。 当 用 户 单 击 “ 编 辑 ” 按 钮 后 ， 即 可 实现 留言 信息 的 更 改 ， 并 且 在 关闭 弹出 窗口 的 瞬间 ， 留 言 信息 的 显 
示 页 面 也 会 自动 进行 刷新 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\14\editleaveword.php) 

<?php 

include_once("conn.php"); 

$id=$_GET["id"]; 

if($_POST["submit"]!="X{ // 判 断 用 户 是 否 单 击 了 “提交 ”按钮 ， 如 果 是 则 执行 如 下 用 于 实现 用 户 留 言 信息 更 改 的 

代码 ， 并 刷新 父 窗口 和 关闭 弹出 窗口 
if(MySQL_query("update tb_leaveword set title=".$_POST["title"].",content=".$_POST["content"].” where 
id=".$_POST["id"]."™",$conn)){ 
echo "<script>alert(' 留 言 更 改 成 功 ! ');window.opener.location.reload();window.close();</script>"; 
}else{ 
echo "<script>alert(' 留 言 更 改 失 败 ! ');history.back();</script>"; 
上 


exit; 


} 
$sql=MySQL_query("select * from tb_leaveword where id=".$id."",$conn); 
// 如 果 用 户 未 单 击 “ 编 辑 ” 按 钮 ， 则 查询 该 用 户 的 留言 内 容 ， 并 最 终 将 未 编辑 的 留言 内 容 显示 在 表单 中 
$info=MySQL_fetch_array($sql); 
?> 


编辑 用 户 留 言 模块 ， 用 户 实现 关闭 弹出 窗口 ， 自 动 刷 新 父 窗口 ， 其 通过 如 下 方法 完成 。 

window.opener.location.reload(); 

上 述 代码 的 实现 原理 是 在 用 window.close0 语 句 关闭 弹出 窗口 前 , 调用 父 窗口 的 reload(0) 方 法 实现 父 窗口 
的 刷新 。 


14.10 ”查询 留言 模块 设计 


14.10.1 查询 留言 模块 概述 


查询 留言 是 对 数据 库 中 的 数据 按 条 件 进行 筛选 浏览 。 本 留言 本 提供 按 主题 、 内 容 、 留 言 者 3 种 条 件 ， 
用 户 可 以 任 选 其 一 作为 查询 条 件 进行 快速 查询 。 查 询 留言 页 面 的 运行 结果 如 图 14.15 所 示 。 


全 
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图 14.15 查询 留言 页 面 的 运行 结果 


14.10.2 ”通过 mysql_fetch_array() 函 数 返 回 结果 集 


PHP 中 提供 了 一 组 操作 MySQL 数据 库 的 函数 ， 应 用 这 些 函 数 可 以 方便 地 对 MySQL 数据 库 进 行 管理 。 
mysql_fetch_array0 函 数 的 作用 是 获得 数据 库 中 满足 mysql_query0 函 数 中 的 SQL 语句 的 记录 , 其 返回 值 
是 一 个 数组 ， 该 数组 的 下 标 可 以 是 字段 名 ， 也 可 以 是 索引 下 标 ， 数 组 元 素 的 值 是 某 个 字段 的 内 容 。 该 函数 
会 使 记录 指针 自动 向 下 移动 ， 当 移动 到 最 后 一 行将 返回 一 个 False 值 。 其 语法 如 下 : 
array mysql_fetch_array(int result,int[result_type]) 
result: 必 选 参数 。mysql_query0 函 数 向 MySQL 数据 库 发 送 查 询 命令 的 返回 标识 。 
result_type: 可 选 参数 。 如 果 该 参数 取 MYSQL _BOTH， 则 该 函数 将 返回 一 个 同时 包含 关联 索引 和 
数字 索引 的 数组 ;如果 该 参数 取 MYSQL _ASSOC， 则 返回 一 个 关联 索引 的 数组 : 如果 取 
MYSQL NUM， 则 返回 一 个 数字 索引 的 数组 。 


开胃 与 该 函数 功能 类 似 的 函数 还 有 MySQL fetch_ rows()、MySQL result0、MySQL fetch_object0 
等 ，MySQL fetch rows(0) 函 数 返回 的 数组 的 下 标 为 数值 索引 下 标 。 MySQL resultO 函 数 有 两 个 参数 ， 其 
中 第 一 个 参数 也 是 MySQL query0 的 返回 结果 ， 而 第 二 个 参数 可 以 是 字段 的 偏 移 量 或 者 字段 名 ， 一 定 注 
意 它 返回 的 结果 不 是 数组 , 而 是 MYSQL 结果 集中 一 个 单元 的 内 容 。MySQL _fetch_objectO 函 数 的 返回 结 
果 是 个 对 象 ， 使 用 时 只 能 通过 字段 名 来 返回 结果 。 


14.10.3 ”查询 留言 模块 实现 过 程 


当 用 户 按 要 求 添加 完 查 找 关 键 字 ， 并 选择 了 查找 方式 对 表单 进行 提交 后 ， 程 序 将 对 用 户 的 提交 内 容 进 
行 判 断 ， 并 最 终 显 示 查 询 结 果 。 实 现 该 过 程 的 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\l4\searchword.php) 
<?php 
if($_POST["submit"]!=™"X{ /判断 用 户 是 否 单 击 了 “查询 ”按钮 ， 如 果 是 则 开始 查询 工作 
$type=$_POST["type"]; 
$keyword=$_POST["keyword"]; 
if($type==1X{ // 根 据 用 户 提交 的 查询 类 型 ， 定 义 查询 语句 
$sql=MySQL_query("select * from tb_leaveword where title like '%".$keyword."%", $conn); 
}elseif($type==2X{ 
$sql=MySQL_query("select * from tb_leaveword where content like '%".$keyword."%",$conn); 
}elseif($type==3X{ 
$sql0=MySQL_query("select id from tb_user where usernc=".$keyword."",$conn); 
$info0=MySQL fetch_array($sql0); 
$sql=MySQL_query("select * from tb_leaveword where userid=".$info0["id"]."",$conn); 
人 


第 14 章 ”综合 实例 (二 ) 一 一 留言 本 


} 
$info=MySQL_fetch_array($sql); 


if($info==false}{ /判断 是 否 查 找到 相关 内 容 ， 如 果 没 查找 到 则 显示 提示 信息 
echo "<br><br><div align=center> 对 不 起 ， 没 有 查找 到 您 要 查找 的 内 容 ! </div>"; 
Jelse{ 
dof // 通 过 循环 显示 查询 结果 
?> 
ee /显示 查找 内 容 
<?phl 


}while($info=MySQL_fetch_array($sql)); 


14.11 版 主 模 块 设计 


医 4 视频 讲解 :光盘 \TM\Video\ 第 14 章 \ 版 主 模块 设计 .exe 


14.11.1 版 主 模块 概述 


为 了 更 好 地 管理 和 维护 留言 本 ， 针 对 留言 本 设置 了 一 个 管理 员 ， 该 管理 员 不 在 后 台 进 行 操作 ， 而 是 在 
前 台 为 管理 员 设 置 特殊 的 权限 ， 也 可 以 称 之 为 版 主 。 其 实现 的 原理 是 : 需要 在 MySQL 数据 库 中 ， 手 动 添加 
-条 系统 版 主 的 数据 ， 版 主 想 要 对 留言 本 进行 管理 时 ， 只 需 单 击 首页 上 版 主管 理 超 链接 ， 登 录 后 即 可 。 版 
主 登 录 界面 如 图 14.16 所 示 ， 版 主 浏览 如 图 14.17 所 示 。 


Ti EE 
11-01-19 D3:38 0 


图 14.16 版 主 登录 界面 图 14.17 版 主 浏览 界面 
14.11.2 ”验证 登录 用 户 是 否 是 版 主 


版 主 单 击 首页 上 的 版 主管 理 进入 版 主 登 录 界 面 。 在 版 主 登录 模块 中 ， 首 先 判断 版 主 名 称 和 密码 是 否 为 
空 ， 然 后 获取 表单 提交 的 值 ， 与 数据 库存 储 的 值 进行 比较 ， 判 断 提交 的 版 主 名 称 与 密码 是 否 与 数据 库 中 的 
版 主 名 称 与 密码 相对 应 ， 如 果 是 则 登录 成 功 ， 反 之 则 登录 失败 。 

版 主 成 功 登 录 后 ， 可 以 对 用 户 提交 的 留言 进行 管理 操作 ， 例 如 ， 全 主题 删除 、 删 除 回复 留言 、 对 用 户 
的 留言 进行 回复 等 。 所 谓 全 主题 删除 ， 主 要 是 删除 留言 及 该 留言 的 回复 信息 。 

其 操作 原理 是 : 当 版 主 单 击 删除 按钮 后 ， 将 弹出 一 个 对 话 框 提 示 版 主 是 否 真正 删除 该 条 留言 及 回复 ， 如 
果 用 户 单 击 对 话 框 中 的 “确定 ”按钮 ， 则 该 条 留言 及 回复 将 被 删除 ， 反 之 不 做 任何 操作 。 其 主要 代码 如 下 


<?php 
国 


$sql="select* from tb_user where id=".$rows['userid].""; 
Sres=mysql_query($sql,$conn); 
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$rew=mysql_fetch_array($res); 
echo unhtml($rew[usernc]); 
?> 
// 表 单元 素 
<?php 
$sqls="select * from tb_replyword where leave_id=".$rows[id].""; 
$rs=mysql_query($sqls,$conn); 
$rw=mysql_num_rows($rs); 
echo $rw; 
?> 
<a href="javascript:if(window.confirm(' 确 定 删除 该 留言 信息 么 ? ')==true){window.location.href='deleteleaveword.php? 
del_id=<?php echo $rows[id]; ?>":}"> 删 除 
ee // 表 单元 素 


14.11.3 ”版 主管 理 模 块 实现 过 程 


为 了 保持 留言 本 的 正常 运行 ， 防止 操作 权限 的 扩大 ， 在 版 主 登 录 处 理 页 中 将 使 用 JavaScript 脚本 限制 用 
户 与 管理 员 在 同一 台 计 算 机 上 登录 的 情况 。 版 主 登录 处 理 页 的 具体 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\14\chklogin.php) 
<?php 
session_start(); 
include("conn/conn.php"); 
if(isset($_SESSION[unc])X{ 
echo "<script>alert(' 在 同一 台 机 器 上 ， 不 允许 同时 使 用 用 户 名 和 管理 员 进 行 登录 ! '); window.location.href= 
'index.php';</script>"; 
}else{ 
ifisset($_POST[Submit]) && $_POST[Submit]==" 登 录 "){ 
if($_POST[username']!="" && $_POST[password"]!="" 
Scheck="select userword from tb_adm where userword=".$_POST[username']."and 
password=".$_POST[password]."™"; 
Sresult=mysql_query($check,$conn); 
Sinfo=mysql_num_rows($result); 
if($info==1X 
$_SESSION["userword"]=$_POST[usemame'; 
echo "<script>alert(' 登 录 成 功 '); window.location.href='admin_browse.php';</script>"; 
jelse{ 
echo "<script>alert(' 登 录 失 败 ， 不 是 版 主 账号 '); window.location.href='index.php';</script>"; 


1 
} 

1 
?> 
当 版 主 确认 删除 全 主题 留言 后 ， 将 进入 全 主题 删除 处 理 页 面 对 全 主题 留言 进行 删除 。 实 现 全 主题 删除 

的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\14\guestbook\deleteleaveword.php) 

<?php 
session_start(); /启用 session 支持 
include_once("conn/conn.php"); // 包 含 数据 库 链 接 文件 


ifisset($_SESSIONTuserword]) && isset($_GET["del_id"]))i{ ”// 对 用 户 进行 判断 ， 是 管理 员 则 执行 全 主题 删除 
Sresult=mysql_query("delete from tb_leaveword where id=".$_GET["del_id"]."",$conn); 
/通过 id 传递 的 值 ， 删 除 留言 表 中 的 留言 
@ 
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S$res=mysql_query("select * from tb_replyword where leave_id=".$_GET["del_id"]."",$conn); 
/查询 回复 表 中 对 该 主题 的 回复 
if(mysql_num_rows($res)>0X // 当 回复 表 中 的 记录 数 大 于 0 了 时， 查询 回复 表 中 对 该 主题 的 回复 
Sresults=mysql_query("delete from tb_replyword where leave_id=".$_GET["del_id"]."",$conn); 
if($result==true and $results==trueX{ // 当 主题 和 回复 都 存在 的 了 时， 执行 全 主题 删除 
echo "<script>alert( 留言 删除 成 功 ! ');history.back();</script>"; 
jelsef 
echo "<script>alert(' 留 言 删除 失败 ! ");history.back();</script>"; 


虽 
Jelse{ 
if($resultX{ /删除 发 布 留言 
echo "<script>alert( 留言 删除 成 功 ! ');history.back();</script>"; 
Jelse{ 
echo "<script>alert(' 留 言 删 除 失败 ! ');history.back();</script>"; 
} 


} 
Jelse{ 
echo "<script>alert(' 您 没有 删除 权限 ! ');history.back();</script>"; 
加 
stm 
14.12 小 结 


本 章 主 要 对 留言 本 的 设计 思路 和 业务 流程 进行 介绍 ， 并 详细 介绍 了 PHP 连接 和 管理 MySQL 数据 库 的 
方法 及 留言 本 的 各 模块 的 实现 过 程 。 通 过 本 章 的 学 习 ， 不 仅 可 以 提高 读者 自身 的 基础 ， 而 且 可 以 全 面 掌 握 
适合 于 中 小 型 企业 应 用 的 留言 本 的 制作 方法 , 最 终 可 以 在 此 基础 上 进行 延伸 , 制作 出 更 有 规模 的 Web 项 目 。 
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MySQL 事务 

触发 中 

综合 实例 ( 三 ) 一 一 物流 管理 系统 
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MySQL 存储 过 程 和 遂 数 


( 铝 | 视频 讲解 : 32 分钟 ) 


存储 过 程 和 函数 是 在 数据 库 中 定义 一 些 SQL 语句 的 集合 ， 然 后 直 
接 调 用 这 些 存 储 过 程 和 函数 来 执行 已 经 定义 好 的 SQL 语句 。 和 存储 过 程 
和 函 教 可 以 避免 开发 人 员 重 复 编写 相同 的 SQL 语句。 而 有 全， 存储 过 程 
和 函数 是 在 My5QL 服务 器 中 存储 和 执行 的 。 可 以 减少 客户 端 和 服务 
器 端的 数据 传输 。 本 章 将 介绍 存储 过 程 和 函 教 的 含义 、 作 用 ， 还 可 以 
了 人 解 创建 、 使 用 、 查 看 、 修 改 及 删除 存储 过 程 及 函数 的 方法 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

M ”了解 流程 控制 语句 的 使 用 

WH 了 解 MySQL 存储 过 程 和 图 数 中 光标 的 使 用 和 一 般 步骤 

H 掌握 MySQL 中 存储 过 程 和 存储 困 数 的 创建 

Wm 掌握 MySQL 存储 过 程 应 用 图 数 的 参数 使 用 方法 

Wm 掌握 存储 过 程 和 图 数 的 调用 、 查 看 、 修 改 和 删除 
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15.1 创建 存储 过 程 和 存储 函数 


在 数据 库 系统 中 ， 为 了 保证 数据 的 完整 性 、 一 致 性 ， 同 时 也 为 提高 其 应 用 性 能 ， 大 多 数据 库 常 采用 存 
储 过 程 和 存储 函数 技术 。MySQL 在 5.0 版 本 后 ， 也 应 用 了 存储 过 程 和 存储 函数 ， 存 储 过 程 和 存储 函数 经 常 
是 一 组 SQL 语句 的 组 合 ， 这 些 语句 被 当 作 整 体 存 入 MySQL 数据 库 服务 器 中 。 用 户 定义 的 存储 函数 不 能 用 
于 修改 全 局 库 状 态 ， 但 该 函数 可 从 查询 中 被 唤醒 调用 ， 也 可 以 像 存储 过 程 一 样 通过 语句 执行 。 随 着 MySQL 
技术 的 日 趋 完善 ， 存 储 过 程 将 和 存储 函数 在 以 后 的 项 目 中 被 到 广泛 应 用 。 


15.1.1 创建 存储 过 程 


铬 il 视频 讲解 : 光盘 \TM\Video\ 第 15 章 \ 视 频 15.1\ 创 建 存储 过 程 .exe 
在 MySQL 中 ， 创 建 存储 过 程 的 基本 形式 如 下 : 
CREATE PROCEDURE sp_name ([proc_parameter[,...]]) 
[characteristic ...] routine_body 
其 中 ，sp_name 参数 是 存储 过 程 的 名 称 ，proc_parameter 表示 存储 过 程 的 参数 列表 ; characteristic 参数 指定 
存储 过 程 的 特性 ，routine_body 参数 是 SQL 代码 的 内 容 ， 可 以 用 begin…end 来 标识 SQL 代码 的 开始 和 结束 。 
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[a 技巧 proc_parameter 中 的 参数 由 3 部 分 组 成 ， 它 们 分 别 是 输入 输出 类 型 、 参 数 名 称 和 参数 类 型 。 其 
形式 为 [IN| OUT |INOUT ljparam name type。 其 中 ，IN 表示 输入 参数 ; OUT 表示 输出 参数 ; INOUT 表 
示 上 既 可 以 输入 也 可 以 输出 ; param_name 参数 是 存储 过 程 参 数 名 称 ; type 参数 指定 存储 过 程 的 参数 类 型 ， 
该 类 型 可 以 为 MySQL 数据 库 的 任意 数据 类 型 。 


-个 存储 过 程 包括 名 字 、 参 数列 表 ， 还 可 以 包括 很 多 SQL 语句 集 。 下 面 创建 一 个 存储 过 程 ， 其 代码 如 下 : 
delimiter 
create procedure proc_name (in parameter integer) 
begin 
declare variable varchar(20); 
if parameter=1 then 
set variable='MySQL'; 
else 
set variable="PHP’; 
end 让 
insert into tb (name) values (variable); 
end; 


MySQL 中 存储 过 程 的 建立 以 create procedure 关键 字 开 始 ， 后 面 仅 跟 存储 过 程 的 名 称 和 参数 。MySQL 
的 存储 过 程 名 称 不 区 分 大 小 写 ， 如 PROCE10 和 proce10 代 表 同 一 存储 过 程 名 。 存 储 过 程 名 或 存储 函数 名 不 
能 与 MySQL 数据 库 中 的 内 建 函 数 重 名 。 

MySQL 存储 过 程 的 语句 块 以 begin 开始 ， 以 end 结束 。 语句 体 中 可 以 包含 变量 的 声明 、 控 制 语句 、SQL 
查询 语句 等 。 由 于 存储 过 程 内 部 语句 要 以 分 号 结束 ， 所 以 在 定义 存储 过 程 前 ， 应 将 语句 结束 标志 “;” 更 改 为 


其 他 字符 ， 并 且 应 降低 该 字符 在 存储 过 程 中 出 现 的 几率 ， 更 改 结束 标志 可 以 用 delimiter 关键 字 定义 。 例 如 : 
mysql>delimiter // 


存储 过 程 创建 后 ， 可 用 如 下 语句 进行 删除 ， 参 数 proc_name 指 存储 过 程 名 。 


drop procedure proc_name 


_ 加 ) 
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下 面 创 建 一 个 名 称 为 count_ of student 的 存储 过 程 。 首 先 ， 创 建 一 个 名 称 为 students 的 MySQL 数据 库 ， 
然后 创建 一 个 名 为 studentinfo 的 数据 表 。 数 据 表 结构 如 表 15.1 所 示 。 


表 15.1 studentinfo 数据 表 结 构 


类 型 (长 度 ) 额 外 说 了 明 


sid | INTUD auto_increment | 主键 自 增 型 sid 
name | VARCHARGS0) | ”学 生 姓名 
age | vaARcHARGD | 学 生年 龄 
SEX | VARCHARGQ) | 学 生性 别 


联系 电话 


BIGINT(1) 


例 15.1 创建 一 个 名 称 为 count_of student 的 存储 过 程 ， 统 计 studentinfo 数据 表 中 的 记录 数 。 (实例 位 
置 : 光盘 \TM\Instance\15\15.1) 

代码 如 下 : 

delimiter // 

create procedure count_of_student(OUT count_num INT) 

reads sql data 

begin 

select count(*) into count_num from studentinfo; 

end 

Uh 


在 上 述 代码 中 , 定义 一 个 输出 变量 zzELECT 语句 从 studentinfo 
表 中 获取 记录 总 数 ， 最 后 将 结果 传递 给 变量 count_num。 存 储 过 程 
的 执行 结果 如 图 15.1 所 示 。 

代码 执行 完毕 后 , 没有 报 出 任何 出 错 信息 
创建 成 功 。 以 后 就 可 以 调用 这 个 存储 过 程 ， 
程 中 的 SQL 语句 。 


储 函 数 已 经 
会 执行 存储 过 图 15.1 存储 过 程 的 执行 结果 


YL 
说 明 MySQL 中 默认 的 语句 结束 符 为 分 号 , 存储 过 程 中 的 SQL 语句 需要 分 号 来 结束 ,为 了 避免 冲突 ， 
首先 用 “DELIMITER //” 将 MySQL 的 结束 符 设置 为 /。 最 后 再 用 “DELIMITER:;” 来 将 结束 符 恢复 成 分 
号 。 这 与 创建 触发 器 时 是 一 样 的 。 


15.1.2 ”创建 存储 函数 


狠 m 视频 讲解 :光盘 \TM\Video\ 第 15 章 \ 创 建 存储 函数 .exe 
创建 存储 函数 与 创建 存储 过 程 大 体 相同 。 其 创建 存储 函数 的 基本 形式 如 下 : 
CREATE FUNCTION sp_name ([func_parameter[,...]]) 
RETURNS type 
[characteristic ...] routine_body 
创建 存储 函数 的 参数 说 明 如 表 15.2 所 示 。 


表 15.2 创建 存储 函数 的 参数 说 明 


说 有明 
存储 函数 的 名 称 
存储 函数 的 参数 列表 


参数 


sp_name 


fun parameter 
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续 表 
参数 说 有明 
RETURNS type 指定 返回 值 的 类 型 
characteristic 指定 存储 过 程 的 特性 
routine bod: SQL 代码 的 内 容 
func_parameter 可 以 由 多 个 参数 组 成 ， 其 中 每 个 参数 均 由 参数 名 称 和 参数 类 型 组 成 。 其 结构 如 下 : 


param_name type 

其 中 ，param_name 参数 是 存储 函数 的 函数 名 称 ; type 参数 用 于 指定 存储 函数 的 参数 类 型 。 该 类 型 可 以 
是 MySQL 数据 库 所 支持 的 类 型 。 

例 15.2 同样， 应 用 studentinfo 表 ， 创 建 名 为 name_of student 的 存储 函数 。 实例 位 置 ， 光盘 \TM 
Instance\15\15.2) 

其 代码 如 下 : 

delimiter // 

create function name_of_student(std_id INT) 

returns varchar(50) 

begin 

return(select name from studentinfo where sid=std_id); 

end 

I 


上 述 代码 中 ， 存 储 函 数 的 名 称 为 name_of _student， 该 函数 
的 参数 为 std_id， 返 回 值 是 VARCHAR 类 型 。 该 函数 实现 从 
studentinfo 表 查 询 与 std_id 相同 sid 值 的 记录 ， 并 将 学 生 名 称 字 
段 name 中 的 值 返回 。 存 储 函数 的 执行 结果 如 图 15.2 所 示 。 


15.1.3 ”变量 的 应 用 


图 15.2 创建 name_of student0 存 储 函数 


名 视频 讲解 : 光盘 \TM\Video\ 第 15 章 \ 变 量 的 应 用 .exe 

MySQL 存储 过 程 中 的 参数 主要 有 局 部 参数 和 会 话 参 数 两 种 ， 这 两 种 参数 又 可 以 被 称 为 局 部 变量 和 会 话 
变量 。 局 部 变量 只 在 定义 该 局 部 变量 的 begin…end 范围 内 有 效 ， 会 话 变 量 在 整个 存储 过 程 范围 内 均 有 效 。 

1. 局 部 变量 

局 部 变量 以 declare 关键 字 声 明 ， 后 跟 变 量 名 和 变量 类 型 ， 例 如 : 

declare a int 

当然 在 声名 局 部 变量 时 也 可 以 用 default 关键 字 为 变量 指定 默认 值 ， 例 如 : 

declare a int default 10 

下 面 的 代码 为 读者 展示 如 何在 MySQL 存储 过 程 中 定义 局 部 变量 以 及 其 使 用 方法 。 在 该 例 中 ,分别 在 内 
层 和 外 层 begin…end 块 中 都 定义 同名 的 变量 x， 按 照 语句 从 上 到 下 执行 的 顺序 ， 如 果 变 量 x 在 整个 程序 中 
都 有 效 ， 则 最 终结 果 应 该 都 为 nner， 但 真正 的 输出 结果 却 不 同 ， 这 说 明 在 内 部 begin…end 块 中 定义 的 变量 
只 在 该 块 内 有 效 。 

例 15.3 该 例 说 明 局 部 变量 只 在 某 个 begin…end 块 内 有 效 。 (实例 位 置 : 光盘 \TMNInstance\1S\15.3) 


代码 如 下 : 

delimiter // 

create procedure p1() 

begin 

declare x char(10) default 'outer '; 
begin 
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declare x char(10) default 'inner '; 
select x; 
end; 
select x; 
end; 
I 
上 述 代码 的 运行 结果 如 图 15.3 所 示 。 
应 用 MySQL 调用 该 存储 过 程 的 运行 结果 如 图 15.4 所 示 。 


图 15.3 定义 局 部 变量 的 运行 结果 图 15.4 调用 存储 过 程 pl0 的 运行 结果 

2. 全 局 变量 

MySQL 中 的 会 话 变量 不 必 声 明 即 可 使 用 ， 会 话 变 量 在 整个 过 程 中 有 效 ， 会 话 变量 名 以 字符 “@” 作 为 
起 始 字 符 。 下 面 的 代码 为 会 话 变 量 的 使 用 方法 。 

例 15.4 在 该 例 中 ， 分别 在 内 部 和 外 部 begin…end 块 中 都 定义 了 同名 的 会 话 变量 @t， 并 且 最 终 输出 结 
果 相 同 ， 从 而 说 明 会 话 变量 的 作用 范围 为 整个 程序 。 (实例 位 置 ， 光盘 \ITM\Instance\15\15.4) 

设置 全 局 变量 的 代码 如 下 : 

delimiter // 

create procedure p2() 

begin 

set @t=1; 

begin 

set @t=2; 

select @t; 

end; 

select @t; 

end; 

I 

上 述 代码 的 运行 结果 如 图 15.5 所 示 。 

应 用 MySQL 调用 该 存储 过 程 的 运行 结果 如 图 15.6 所 示 。 


图 15.5 设置 全 局 变量 图 15.6 调用 存储 过 程 p20 运行 结 
3. 为 变量 赋值 
MySQL 中 可 以 使 用 DECLARE 关键 字 来 定义 变量 。 定 义 变量 的 基本 语法 如 下 : 
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DECLARE var_name[,...] type [DEFAULT value] 

其 中 ，DECLARE 是 用 来 声明 变量 的 ，var_name 参数 是 设置 变量 的 名 称 。 如 果 用 户 需要 ， 也 可 以 同时 
定义 多 个 变量 。type 参数 用 来 指定 变量 的 类 型 ， DEFAULT value 的 作用 是 指定 变量 的 默认 值 ， 不 对 该 参数 
进行 设置 时 ， 其 默认 值 为 NULL。 

MySQL 中 可 以 使 用 SET 关键 字 为 变量 赋值 。SET 语句 的 基本 语法 如 下 : 

SET var_name=expr[,var_name=expr]... 

其 中 ,SET 关键 字 是 用 来 为 变量 赋值 : var_name 参数 是 变量 的 名 称 ; expr 参数 是 赋值 表达 式 。 一 个 SET 
语句 可 以 同时 为 多 个 变量 赋值 ， 各 个 变量 的 赋值 语句 之 间 用 “,” 隔 开 。 例如， 为 变量 mr soft 赋值 ， 代 码 如 下 : 

SET mr_soft=10; 

另外 MySQL 中 还 可 以 应 用 另 一 种 方式 为 变量 赋值 。 其 语法 结构 如 下 : 

SELECT col_namel[,...] INTO var_namel,...] FROM table_name where condition 

其 中 ，col name 参数 标识 查询 的 字段 名 称 ; var_name 参数 是 变量 的 名 称 ，table_name 参数 为 指定 数据 
表 的 名 称 ，condition 参数 为 指定 查询 条 件 。 例 如 ， 从 studentinfo 表 中 查询 name 为 LeonSK 的 记录 。 将 该 记 
录 下 的 tel 字段 内 容 赋值 给 变量 customer_tel。 其 关键 代码 如 下 : 

SELECT tel INTO customer_tel FROM studentinfo WHERE name= 'LeonSK '; 


/ 
后 明 上 述 赋值 语句 必须 存在 于 创建 的 存储 过 程 中 。 且 需 将 赋值 语句 放置 在 BEGIN.…END 之 间 。 若 
脱离 此 范围 ， 该 变量 将 不 能 使 用 或 被 赋值 。 


15.1.4 ”光标 的 运用 


铭 4 视频 讲解 : 光盘 \TM\Video\ 第 15 章 \ 光 标的 运用 .exe 

通过 MySQL 查询 数据 库 ， 其 结果 可 能 为 多 条 记录 。 在 存储 过 程 和 函数 中 使 用 光标 可 以 实现 逐条 读 取 结 
果 集中 的 记录 。 光 标 使 用 包括 声明 光标 (DECLARE CURSOR) 、 打 开光 标 (OPEN CURSOR) 、 使 用 光标 
(FETCH CURSOR) 和 关闭 光标 〈CLOSE CURSIR) 。 值 得 一 提 的 是 ， 光 标 必 须 声 明 在 处 理 程序 之 前 ， 且 
声明 在 变量 和 条 件 之 后 。 

1. 声明 光标 

在 MySQL 中 ， 声 明光 标 仍 使 用 DECLARE 关键 字 。 其 语法 如 下 : 

DECLARE cursor_name CURSOR FOR select_statement 

其 中 ，cursor_name 是 光标 的 名 称 ， 光 标 名 称 使 用 与 表 名 同样 的 规则 ;select_statement 是 一 个 SELECT 
语句 ， 返 回 一 行 或 多 行 数据 。 其 中 这 个 语句 也 可 以 在 存储 过 程 中 定义 多 个 光标 ， 但 是 必须 保证 每 个 光标 名 
称 的 唯一 性 ， 即 每 一 个 光标 必须 有 自己 唯一 的 名 称 。 

通过 上 述 定义 来 声明 光标 info_of student， 其 代码 如 下 : 

DECLARE info_of_student CURSOR FOR SELECT 

sid,name,age,sex,age 

FROM studentinfo 

WHERE sid=1; 


\ 


E 技巧 这 里 SELECT 子 句 中 不 能 包含 INTO 子 句 ， 并 且 光 标 只 能 在 存储 过 程 或 存储 函数 中 使 用 。 上 
述 代 码 并 不 能 单独 执行 。 

2. 打开 光标 

在 声明 光标 之 后 ， 要 从 光标 中 提取 数据 ， 必 须 首 先 打开 光标 。 在 MySQL 中 ， 使 用 OPEN 关键 字 来 打开 


@ 
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光标 。 其 基本 的 语法 如 下 : 

OPEN cursor_name 

其 中 ，cursor_name 参数 表示 光标 的 名 称 。 在 程序 中 ， 一 个 光标 可 以 打开 多 次 。 由 于 可 能 在 用 户 打开 光 
标 后 ， 其 他 用 户 或 程序 正在 更 新 数据 表 。 所 以 可 能 会 导致 用 户 在 每 次 打开 光标 后 ， 显 示 的 结果 都 不 同 。 

打开 上 面 已 经 声明 的 光标 info_of student， 其 代码 如 下 : 

OPEN info_of_student 

3. 使 用 光标 

光标 在 顺利 打开 后 ， 可 以 使 用 FETCH…INTO 语句 来 读 取 数据 。 其 语法 如 下 : 

FETCH cursor_name INTO var_name[,var_name]… 

其 中 ，cursor_ name 代表 已 经 打开 的 光标 名 称 ; var_name 参数 表示 将 光标 中 的 SELECT 语句 查询 出 来 的 
信息 存 入 该 参数 中 。var_name 是 存放 数据 的 变量 名 ， 必 须 在 声明 光标 前 定义 好 。FETCH…INTO 语句 与 
SELECT…INTO 语句 具有 相同 的 意义 。 

将 已 打开 的 光标 info_of student 中 SELECT 语句 查询 出 来 的 信息 存 入 tmp_name 和 tmp_tel 中 。 其 中 
tmp_name 和 tmp_tel 必须 在 使 用 前 定义 。 其 代码 如 下 : 

FETCH info_of_student INTO tmp_name,tmp_tel; 

4. 关闭 光标 

光标 使 用 完毕 后 ， 要 及 时 关闭 ， 在 MySQL 中 采用 CLOSE 关键 字 关 闭 光 标 。 其 语法 格式 如 下 : 

CLOSE cursor_name 

cursor_name 参数 表示 光标 名 称 。 

下 面 关闭 已 打开 的 光标 info_of student。 代 码 如 下 : 

CLOSE info_of_student 


和 


| 完毕 后 一 定 要 
关闭 。 


15$.2 ”流程 控制 语句 


在 MySQL 中 ， 常 见 的 过 程式 SQL 语句 可 以 用 在 一 个 存储 过 程 体 中 。 其 中 包括 正 语句 、CASE 语句 、 
LOOP 语句 、WHILE 语句 、ITERATE 语句 和 LEAVE 语句 ， 它 们 可 以 进行 流程 控制 。 


15.2.1 IF 语句 


正 语句 用 来 进行 条 件 判断 ， 根 据 不 同 的 条 件 执行 不 同 的 操作 。 该 语句 在 执行 时 首先 判断 正 后 的 条 件 是 
否 为 真 ， 如 果 为 真 则 执行 THEN 后 的 语句 ， 如 果 为 假 则 继续 判断 正 语句 直到 为 真 为 止 ， 当 以 上 都 不 满足 时 
则 执行 ELSE 语句 后 的 内 容 。 正 语句 表示 形式 如 下 : 

IF condition THEN 

[ELSE condition THEN] 

[ELSE] 


ENDIF 


@ 
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例 15.5 下 面 通过 if…then*…else 结构 首先 判断 传 入 参数 的 值 是 否 为 1， 如 果 是 则 输出 1， 如 果 不 是 则 
再 判断 该 传 入 参数 的 值 是 否 为 2， 如 果 是 则 输出 2， 当 以 上 条 件 都 不 满足 时 输出 3。 (实例 位 置 : 光盘 \TM 
Instance\15\15.5) 

其 代码 如 下 : 

delimiter // 

create procedure example_if(in x int) 

begin 

if x=1 then 

select 1; 

elseif x=2 then 

select 2; 

else 

select 3; 

end if, 

end 

I 

以 上 代码 的 运行 结果 如 图 15.7 所 示 。 

通过 MySQL 调用 该 存储 过 程 ， 其 运行 结果 如 图 15.8 所 示 。 


图 15.7 应 用 正 语句 的 存储 过 程 图 15.8 调用 example if 存储 过 程 


15.2.2 CASE 语句 


CASE 语句 为 多 分 支 语句 结构 ， 该 语句 首先 从 when 后 的 value 中 查找 与 case 后 的 value 相等 的 值 ， 如 
果 查 找到 则 执行 该 分 支 的 内 容 ， 否 则 执行 else 后 的 内 容 。CASE 语句 表示 形式 如 下 : 
CASE value 
WHEN value THEN… 
[WHEN valueTHEN…] 
[ELSE.…] 
END CASE 
其 中 ，value 参数 表示 条 件 判 断 的 变量 ，WHEN…THEN 中 的 value 参数 表示 变量 的 取 值 。 
CASE 语句 还 有 另 一 种 语法 表示 结构 : 
CASE 
WHEN value THEN… 
[WHEN valueTHEN…] 
[ELSE.…] 
END CASE 


| 区 技巧 


一 个 CASE 语句 经 常 可 以 充当 一 个 正 …THEN…ELSE 语句 。 


例 15.6 下 面 通过 CASE 语句 首先 判断 传 入 参数 的 值 是 否 为 1， 如果 条 件 成 立 则 输出 1， 如 果 条 件 不 成 
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立 则 再 判断 该 传 入 参数 的 值 是 否 为 2， 如 果 成 立 则 输出 2， 当 以 上 条 件 都 不 满足 时 输出 3。 (实例 位 置 : 光 
盘 \TMNVInstance\1S\15.6) 
代码 如 下 : 


delimiter // 

create procedure example_case(in x int) 
begin 

casex 

when 1 then select 1; 

when 2 then select 2; 

else select 3; 

end case; 

end 

I 


运行 该 实例 的 结果 如 图 15.9 所 示 。 
调用 该 存储 过 程 ， 其 运行 结果 如 图 15.10 所 示 。 


图 15.9 应 用 CASE 语句 的 存储 过 程 图 15.10 调用 example_case0 存 储 过 程 
15.2.3 WHILE 循环 语句 


WHILE 循环 语句 执行 时 首先 判断 condition 条 件 是 否 为 真 ， 如 果 是 则 执行 循环 体 ， 否 则 退出 循环 。 该 语 
句 表示 形式 如 下 : 
while condition do 


end while; 

例 15.7 下 面 应 用 WHILE 语句 求 前 100 项 的 和 。 首 先 定 义 变量 i 和 s， 分 别 用 来 控制 循环 的 次 数 和 保 
存 前 100 项 的 和 ， 当 变量 i 的 值 小 于 或 等 于 100 时 ， 使 s 的 值 加 i， 并 同时 使 i 的 值 增 1。 直 到 i 大 于 100 时 
退出 循环 并 输出 结果 。 (实例 位 置 ; 光盘 \TMNInstance\1S\15.7) 


代码 如 下 : 

delimiter // 

create procedure example_while (out sum int) 
begin 

declare i int default 1; 
declare s int default 0; 
while i<=100 do 

Set s=s+i; 

set i=i+1; 

end while; 

set sum=s; 

end 

I 


运行 以 上 代码 的 结果 如 图 15.11 所 示 。 
调用 该 存储 过 程 ， 调 用 语句 如 下 : 
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call example_while(@s) 
加 9 @s 


调用 该 存储 过 程 的 结果 如 图 15.12 所 示 。 


图 15.11 应 用 WHILE 语句 的 存储 过 程 图 15.12 调用 example_while0 存 储 过 程 
15.2.4 LOOP 循环 语句 


该 循环 没有 内 置 的 循环 条 件 ， 但 可 以 通过 LEAVE 语句 退出 循环 。LOOP 语句 表示 形式 : 

loop 

end loop 

LOOP 允许 某 特定 语句 或 语 句 群 的 重复 执行 , 实现 一 个 简单 的 循环 构造 ,其 中 中 间 省 略 的 部 分 是 需要 重 
复 执行 的 语句 。 在 循环 内 的 语句 一 直 重 复 直 至 循环 被 退 ， 0 LEAVE 语句 。 

LEAVE 语句 经 常 和 BEGIN…END 或 循环 一 起 使 用 ， 其 结构 如 下 : 

LEAVE label 

label 是 语句 中 标注 的 名 字 , 这 个 名 字 是 自 定义 的 。 加 上 LEAVE 关键 字 就 可 以 用 来 退出 被 标注 的 循环 
语句 。 

例 15.8 下 面 应 用 LOOP 语句 求 前 100 项 的 和 。 首 先 定义 变量 1 和 s， 分 别 用 来 控制 循环 的 次 数 和 保存 
前 100 项 的 和 , 进入 该 循环 体 后 首先 使 s 的 值 加 i, 之 后 使 1 加 1 并 进入 下 次 循环 , 直到 i 大 于 100, 通过 leave 
语句 退出 循环 并 输出 结果 。〔 实 例 位 置 ， 光盘 \TM\Instance\15\15.8) 


代码 如 下 : 

delimiter // 

create procedure example_loop (out sum int) 
begin 

declare i int default 1; 
declare s int default 0; 
loop_label:loop 

Set s=Ss+i; 

Set i=i+1; 

ifi>100 then 

leave loop_label; 

end if: 

end loop; 

set sum=s; 

end 

上 


述 代码 的 运行 结果 如 图 15.13 所 示 。 
名 称 为 example loop 的 存储 过 程 ， 其 代码 如 下 : 


call example_loop(@s) 
select @s 


运行 结果 如 图 15.14 所 示 。 
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图 15.13 ”应 用 LOOP 语句 创建 存储 过 程 


15.2.5 “REPEAT 循环 语句 
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图 15.14 调用 example loop0 存 储 过 程 


该 语句 先 执行 一 次 循环 体 ， 之 后 判断 condition 条 件 是 否 为 真 ， 若 为 真 则 退出 循环 ， 否 则 继续 执行 循环 。 
repeat 语句 表示 形式 如 下 : 


REPEAT 


UNTIL condition 
END REPEAT 


例 15.9 下面 应 用 repeat 语句 求 前 100 项 和 。 首 先 定 义 变量 i 和 s， 分 别 用 来 控制 循环 的 次 数 和 保存 前 
100 项 的 和 ， 进 入 循环 体 后 首先 使 s 的 值 加 i， 之 后 使 1 的 值 加 1， 直 到 i 大 于 100 时 退出 循环 并 输出 结果 。 
(实例 位 置 ， 光盘 \TM\Instance\15\15.9) 


代码 如 下 : 

delimiter // 

create procedure example_repeat (out sum int) 
begin 

declare i int default 1; 

declare s int default 0; 

repeat 

Set s=s+i; 

Set j=i+1; 
until i>100 

end repeat; 

set sum=s; 

end 

Uh 

以 上 代码 的 运行 结果 如 图 15.15 所 示 。 

调用 该 存储 过 程 ， 相 关 代码 如 下 : 

call example_repeat(@s) 

select @s 

调用 该 存储 过 程 的 运行 结果 如 图 15.16 所 示 。 


图 15.15 ”应 用 REPEAT 语句 创建 存储 过 程 


图 15.16 调用 example_repeat0 存 储 过 程 
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循环 语句 中 还 有 一 个 ITERATE 语句 ， 它 可 以 出 现在 LOOP、REPEAT 和 WHILE 语句 内 ， 其 意 为 再 次 
循环 。 该 语句 格式 如 下 : 

ITERATE label 

该 语句 的 格式 与 LEAVE 大 同 小 异 ， 区 别 在 于 LEAVE 语句 是 离开 一 个 循环 ， 而 ITERATE 语句 是 重新 
开始 一 个 循环 。 


人 注意 与 一 般 程 序 设计 流程 控制 不 同 的 是 存储 过 程 并 不 支持 FOR 循环 。 


15.3 ”调用 存储 过 程 和 存储 函数 


铭 4 视频 讲解 : 光盘 \TM\Video\ 第 15 章 \ 调 用 存储 过 程 和 存储 函数 .exe 
存储 过 程 和 存储 函数 都 是 存储 在 服务 器 的 SQL 语句 的 集合 。 要 使 用 这 些 已 经 定义 好 的 存储 过 程 和 存储 
函数 就 必须 要 通过 调用 的 方式 来 实现 。 


15.3.1 调用 存储 过 程 
存储 过 程 的 调用 在 前 面 的 实例 中 多 次 被 用 到 。MySQL 中 使 用 CALL 语句 来 调用 存储 过 程 。 调 用 存储 过 
程 后 ， 数 据 库 系统 将 执行 存储 过 程 中 的 语句 ， 然 后 将 结果 返回 给 输出 值 。CALL 语句 的 基本 语法 形式 如 下 : 
CALL sp_name([parameter[,...]]); 
其 中 ，sp_name 是 存储 过 程 的 名 称 ; parameter 是 存储 过 程 的 参数 。 
15.3.2 ”调用 存储 函数 
在 MySQL 中 ， 存 储 函数 的 使 用 方法 与 MySQL 内 部 函数 的 使 用 方法 基本 相同 。 用 户 自 定义 的 存储 函数 
与 MySQL 内 部 函数 性 质 相同 。 区 别 在 于 存储 函数 是 用 户 自 定义 的 ， 而 内 部 函数 由 MySQL 自 带 。 其 语法 结 


构 如 下 : 
SELECT function_name([parameter[,...]]); 


15.4 查看 存储 过 程 和 函数 


峰 1 视频 讲解 : 光盘 \TM\Video\ 第 15 章 \ 查 看 存储 过 程 和 函数 .exe 

存储 过 程 和 函数 创建 以 后 , 用 户 可 以 查看 存储 过 程 和 函数 的 状态 和 定义 用户 可 以 通过 SHOW STATUS 
语句 查看 存储 过 程 和 函数 状态 ， 也 可 以 通过 SHOW CREATE 语句 来 查看 存储 过 程 和 函数 的 定义 。 
15.4.1 SHOW STATUS 语句 


在 MySQL 中 可 以 通过 SHOW STATUS 语句 查看 存储 过 程 和 函数 的 状态 。 其 基本 语法 结构 如 下 : 
SHOW {PROCEDURE | FUNCTION}STATUSILIKE 'pattern] 
-加 
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其 中 ，PROCEDURE 参数 表示 查询 存储 过 程 ， FUNCTION 参数 表示 查询 存储 函数 ;LIKE '\pattern' 参 数 
用 来 匹配 存储 过 程 或 函数 名 称 。 


15.4.2 SHOW CREATE 语句 


MySQL 中 可 以 通过 SHOW CREATE 语句 来 查看 存储 过 程 和 函数 的 状态 。 其 语法 结果 如 下 : 
SHOW CREATE{PROCEDURE | FUNCTION } sp_name; 
其 中 ，PROCEDURE 参数 表示 存储 过 程 ， FUNCTION 参数 表示 查询 存储 函数 ，sp_name 参数 表示 存储 
过 程 或 函数 的 名 称 。 
例 15.10 ”下 面 查询 名 为 count_of student 的 存储 过 程 。 (实例 位 置 ， 光盘 \TMNInstance\1S\1S.10) 
其 代码 如 下 : 
show create procedure count_of_student ; 
其 运行 结果 如 图 15.17 所 示 。 


图 15.17 应 用 SHOW CREATE 语句 查看 存储 过 程 
查询 结果 显示 存储 过 程 的 定义 、 字 符 集 等 信息 


区 9 技巧 SHOW STATUS 语句 只 能 Wo 如 存储 过 程 或 函数 的 名 
称 、 类 型 、 定 义 者 、 修 改 时 间 等 信息 ， 并 不 能 查询 存储 过 程 或 函数 的 具体 定义 。 如 果 需 要 查看 详细 定义 ， 
需要 使 用 SHOW CREATE 语句 。 


15.5 修改 存储 过 程 和 函数 


医 1 视频 讲解 :光盘 \TM\Video\ 第 15 章 \ 修 改 存 储 过 程 和 函数 .exe 

修改 存储 过 程 和 存储 函数 是 指 修改 已 经 定义 好 的 存储 过 程 和 函数 .MySQL 中 通过 ALTER PROCEDURE 
语句 来 修改 存储 过 程 ， 通 过 ALTER FUNCTION 语句 来 修改 存储 函数 。 

MySQL 中 修改 存储 过 程 和 函数 的 语法 形式 如 下 : 


ALTER {PROCEDURE | FUNCTION} sp_name [characteristic ..] 
characteristic: 


{ CONTAINS SQL | NO SQL | READS SQL DATA | MODIFIES SQL DATA} 
|SQL SECURITY { DEFINER | INVOKER } 
| COMMENT 'string’ 


其 参数 说 明 如 表 15.3 所 示 。 


他 
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表 15.3 ”修改 存储 过 程 和 函数 的 语法 的 参数 说 明 


参数 说 明 
sp_name 存储 过 程 或 函数 的 名 称 
characteristic 指定 存储 函数 的 特性 
CONTAINS SQL 表示 子 程序 包含 SQL 语句 ， 但 不 包含 读 写 数据 的 语句 
NO SQL 表示 子 程序 不 包含 SQL 语句 
READS SQL DATA 表示 子 程序 中 包含 读数 据 的 语句 
MODIFIES SQL DATA 表示 子 程序 中 包含 写 数据 的 语句 
SQL 指明 权限 执行 。DEFINER 表示 只 有 定义 者 自己 才能 够 执行 ，INVOKER 表示 调 
SECURITY{DEFINERIINVOKER} | 用 者 可 以 执行 
COMMENT'string’ 是 注释 信息 


例 15.11 下 面 应 用 此 语句 修改 存储 过 程 count_of student。《〔 实 例 位 置 ， 光 盘 \TM\Instance\15\15.11) 
其 代码 如 下 : 

alter procedure count_of_student 
modifies sql data 

Sql security invoker; 

其 运行 结果 如 图 15.18 所 示 。 


二 
[ES 如 果 读 者 希望 查看 修改 后 的 结果 ， 可 以 应 用 SELECT 、 
FROM studentinfo Ruotines WHERE ROUTINE NAME='sp name 来。 图 15.18 修改 存储 过 程 count_ of_ 
查看 表 的 信息 。 由 于 篇 幅 限 制 ， 这 里 不 能 详细 地 讲解 。 student 的 定义 


15.6 ”删除 存储 过 程 和 函数 


铬 il 视频 讲解 :光盘 \TM\Video\ 第 15 章 \ 删 除 存储 过 程 和 函数 .exe 

删除 存储 过 程 和 函数 指 删除 数据 库 中 已 经 存在 的 存储 过 程 和 函数 。MySQL 中 使 用 DROP PROCEDURE 
语句 来 删除 存储 过 程 ， 通 过 DROP FUNCTION 语句 来 删除 存储 函数 。 在 删除 之 前 ， 必 须 确认 该 存储 过 程 或 
函数 没有 任何 依赖 关系 ， 和 否则 可 能 会 导致 其 他 与 其 关联 的 存储 过 程 无 法 运行 。 

删除 存储 过 程 和 函数 的 语法 如 下 : 

DROP {PROCEDURE | FUNCTION} [IF EXISTS] sp_name 

其 中 ，sp_name 参数 表示 存储 过 程 或 函数 的 名 称 ; IF EXISTS 是 MySQL 的 扩展 ， 判 断 存储 过 程 或 函数 
是 否 存在 ， 以 免 发 生 错误 。 

例 15.12 下面 删除 名 称 为 count_of _ student 的 存储 过 程 。〈 实 例 位 置 : 光盘 \TM\Instance\15\15.12) 

其 关键 代码 如 下 : 

drop procedure count_of_student; 

删除 存储 过 程 count_of student 的 运行 结果 如 图 15.19 所 示 。 

例 15.13 下面 删除 名 称 为 name_of student 的 存储 函数 。 (实例 位 置 : 光盘 \TM\Instance\15\15.13) 

其 关键 代码 如 下 : 

drop function name_of_student; 
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删除 存储 函数 name of student 的 运行 结果 如 图 15.20 所 示 。 


图 15.19 删除 count of student 存储 过 程 15.20 ”删除 name of student 存储 函数 


当 返 回 结果 没有 提示 警告 或 报错 时 ， 则 说 明 存储 过 程 或 存储 函数 已 经 被 顺利 删除 。 用 户 可 以 通过 查询 
students 数据 库 下 的 Routines 表 来 确认 上 面 的 删除 是 否 成 功 。 


15.7 捕获 存储 过 程 中 的 错误 


驹 4 视频 讲解 : 光盘 \TM\Video\ 第 15 章 \ 捕 获 存储 过 程 中 的 错误 .exe 

在 用 户 执行 某 些 程序 时 ， 可 能 会 产生 一 些 问题 ， 为 了 增强 程序 本 身 处 理 问题 的 能 力 ， 避 免 程序 因 异 常 
而 终止 运行 。 我 们 往往 在 处 理 程序 执行 前 ， 预 测 程序 在 执行 过 程 中 可 能 出 现 或 遇 到 的 问题 。 定 义 条 件 和 处 
理 程序 来 提示 用 户 的 同时 , 也 为 用 户 提出 解决 办 法 .MySQL 通过 DECLARE 关键 字 来 定义 条 件 和 处 理 程序 。 


15.7.1 定义 条 件 


在 MySQL 中 ， 应 用 DECLARE 语句 定义 条 件 。 其 语法 格式 如 下 : 

DECLARE condition_name CONDITION FOR condition_value 

condition_value: 

SQLSTATE [VALUE] sqlstate_value | mysql_error_code 

其 中 ，condition_name 参数 表示 条 件 名 称 ; condition_value 参数 表示 类 型 ， sqlstate_value 参数 可 以 表示 
MySQL 的 错误 。 另 外 也 可 以 应 用 mysql_error_code 来 表示 错误 代码 。 其 中 第 一 种 定义 方法 的 格式 如 下 : 

DECLARE can_not_find CONDITION FOR SQLSTATE ‘42S02’; 

第 二 种 定义 格式 如 下 : 

DECLARE can_not_find CONDITION FOR 1146 


p94 
“ 说 明 两 种 表示 MySQL 错误 的 方法 名 称 都 为 can_not find， 不 同 点 只 是 获取 的 错误 提示 码 不 同 。 第 
一 种 方法 设置 sqlstate_value 值 为 42S02， 第 二 种 方法 设置 mysq] error code 值 为 1146。 


15.7.2 ”定义 处 理 程序 


另外 ，MySQL 中 也 可 以 使 用 DECLARE 关键 字 来 定义 处 理 程序 。 其 语法 如 下 : 
DECLARE handler_type HANDLER FOR condition_value[,…] sp_statement 
handler_type: 

CONTINUE 

1EXIT 

1UNDO 
condition_value: 

SQLSTATE [VALUE] sqlstate_value| condition_name | SQLWARNING | NOT FOUND 
| SQLEXCEPTION| mysql_error_code 


这 个 语句 指定 每 个 可 以 处 理 一 个 或 多 个 条 件 的 处 理 程序 。 如 果 产 生 一 个 或 多 个 条 件 ， 指 定 的 语句 被 执行 。 


全 
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加 


handler type: 指明 错误 的 处 理 方式 ， 该 参数 有 3 个 取 值 。 它 们 分 别 是 : 

> ”对 于 一 个 CONTINUE 处 理 程序 ， 当 前 子 程序 的 执行 在 执行 处 理 程序 语句 之 后 继续 。 
> ”对 于 EXIT 处 理 程序 ， 当 前 BEGIN…END 复合 语句 的 执行 被 终止 。 

> UNDO 表示 遇 到 错误 后 撤回 之 前 的 操作 ，MySQL 不 支持 该 处 理 方式 。 

condition value: 指定 错误 类 型 ， 该 参数 有 6 个 取 值 。 

sqlstate_value: 表示 MySQL 的 错误 。 

Imysql_error_ code: 表示 错误 代码 

condition name: DECLARE 定义 的 条 件 名 称 。 

SQL WARNING: 表示 所 有 以 01 开头 的 sqlstate_value 值 。 

NOT FOUND: 表示 所 有 以 02 开头 的 sqlstate_value 值 。 

SQLEXCEPITION: 表示 所 有 没有 被 SQL WARNING 或 NOT FOUND 捕获 的 sqlstate_value 值 。 


因 罗 办 因 办 办 办 


15.8 实 战 


15.8.1 ”使 用 存储 过 程 实现 用 户 注册 (PHP) 


例 15.14 将 向 读者 介绍 MySQL 5.5 版 本 中 存储 过 程 的 创建 以 及 PHP 调用 MySQL 存储 过 程 的 方式 。 
在 图 中 的 文本 框 中 输入 注册 信息 后 ， 单 击 “ 注 册 ” 按 钮 即 可 将 用 户 填写 的 注册 信息 保存 到 数据 库 中 。〔 实 
例 位 置 ， 光盘 \TM\Instance\15\15.14) 

实现 过 程 如 下 所 示 。 

(1) 创建 pro_reg 存储 过 程 ， 其 代码 如 下 : 

delimiter // 

create procedure pro_reg(in nc varchar(50),in pwd varchar(50),in email varchar(50),in address varchar(50)) 

begin 

insert into tb_reg(name,pwd,email,address) values (nc,pwd,email,address); 

end; 

I 


(2) 通过 PHP 预定 义 类 mysqli 实现 与 MySQL 数据 库 的 连接 。 代 码 如 下 : 


<?php 
if($_POSTrsubmit]!="){ 
S$conn=new mysqli("localhost","root","111","db_database15"); /| 连接 数据 库 
S$conn->query("set names utf8"); /设置 编码 格式 


S$name=$_POST[name]; 
$pwd=md5($_POST[pwd]); 
S$email=$_POST[email]; 
$address=$_POST[address']; 
(3) 调用 存储 过 程 pro_reg 实现 将 用 户 录入 的 注册 信息 保存 到 数据 库 中 。 代 码 如 下 : 
if($sql=$conn->query("call pro_reg(".$name.",".$pwd.",".$email.",".$address.")"){ ”// 调 用 存储 过 程 
echo "<script>alert(' 用 户 注册 成 功 "");</script>"; 
}else{ 
echo "<script>alert(' 用 户 注册 失败 ");</script>"; 
} 
} 


2> 


运行 结果 如 图 15.21 一 图 15.23 所 示 。 


@ 
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图 15.21 创建 存储 过 程 


证 出 | 至 号 FX Ti 


图 15.22 录入 注册 信息 图 15.23 注册 信息 被 存储 到 MySQL 数据 库 


15.8.2 ”修改 存储 函数 


例 15.15 将 name_of student 的 存储 函数 的 读 写 权限 修改 为 READS SQL DATA， 并 加 上 注释 信息 
‘FIND NAME ' 。 修改 存储 过 程 和 存储 函数 是 指 修改 已 经 定义 好 的 存储 过 程 和 函数 .MySQL 中 通过 ALTER 
PROCEDURE 语句 来 修改 存储 过 程 。( 实 例 位置 ， 光盘 \TMNInstance\1S\1S.1S) 
代码 如 下 : 
ALTER FUNCTION name_of_student READS SQL DATA COMMENT ‘FIND NAME '; 
SELECT SPCIFIC_NAME,SQL_DATA_ACCESS,ROUTINE_COMMENT FROM information_schema.Routines 
WHERE ROUTINE_NAME='name_of_student'; 


运行 结果 如 图 15.24 所 示 。 
15.8.3 从 information_schema.Routines 表 中 查看 存储 过 程 


存储 过 程 和 函数 的 信息 存储 


例 15.16 从 Routines 表 中 查询 名 为 count_of _ student 的 存储 过 程 的 。 
查询 存储 过 程 和 函数 的 信息 。 


在 information_ schema 数据 库 下 的 Routines 表 中 ， 可 以 通过 查询 该 表 的 记 
(实例 位 置 ， 光 盘 \TM\Instance\15\15.16) 
关键 代码 如 下 : 
SELECT * FROM information_schema.Routines WHERE ROUTINE_NAME='count_of_student \G; 
运行 结果 如 图 15.25 所 示 。 


ROUTINE 0 


PIND NAME 


15.24 ”修改 存储 函数 图 15.25 从 information schema.Routines 表 中 查看 存储 过 程 


| 
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15.9 小 结 


本 章 对 MySQL 数据 库 的 存储 过 程 和 存储 函数 进行 了 详细 讲解 , 存储 过 程 和 存储 函数 都 是 用 户 自 己 定义 
的 SQL 语句 的 集合 。 它 们 都 存储 在 服务 器 端 ， 只 要 调用 就 可 以 在 服务 器 端 执行 。 本 章 重点 讲解 了 创建 存储 
过 程 和 存储 函数 的 方法 ， 通 过 CREATE PROCEDURE 语句 来 创建 存储 过 程 ， 通 过 CREATE FUNCTION 语 
句 来 创建 存储 函数 。 需 要 读者 将 书 中 的 知识 点 结合 实际 操作 进行 练习 。 


15.10 ”学 习 成 果 检 验 


1. 存储 过 程 和 存储 函数 的 区 别 是 什么 ? 〈 实 例 位 置 ， 光盘 \TMNInstance\1S\15.17) 
2. 一 个 存储 过 程 中 可 以 调用 其 他 的 存储 过 程 吗 ?实例 位 置 ， 光盘 \TM\Instance\15\15.18) 
3. 存储 函数 和 MySQL 内 部 函数 有 什么 区 别 ? 《实例 位 置 : 光盘 \ITM\Instance\15\15.19) 


1 人。 


MySQL 事务 
(名 视频 讲解 : 14 分 钟 ) 


在 操作 My5QL 的 过 程 中 ， 对 于 一 般 的 简单 业务 逻辑 或 中 小 型 程 
序 而 言 ， 无 须 考虑 应 用 MySQL 事务 。 但 在 比较 复杂 的 情况 下 ， 在 执 
行 某 些 数据 操作 过 程 中 ,往往 需要 通过 一 组 SQL 语句 执行 多 项 并 行业 
务 远 辑 或 程序 ， 这 样 ， 就 必须 保证 所 用 命令 执行 的 同步 性 ， 使 执行 序 
列 中 产生 依靠 关系 的 动作 能 够 同时 操作 成 功 或 同时 返回 初始 状态 。 在 
此 情况 下 ， 就 需要 用 户 优先 考虑 使 用 My5QL 事务 处 理 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

WM 了 解 MySQL 事务 

WH 了 解 MySQL 事务 的 存在 周期 

mH ”掌握 如 何 创 建 事 务 

”掌握 MySQL 事务 的 查询 、 提 交 和 回 滚 

WH 掌握 MySQL 事务 行为 

WH 掌握 MySQL 事务 的 性 能 

MW 掌握 如 何 使 用 MySQL 的 伪 事 务 
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16.1 MySQL 事务 概述 


区 4 视频 讲解 : 光盘 \TM\Video\ 第 16 章 \MySQL 事务 概述 .exe 

在 MySQL 中 , 事务 由 单独 单元 的 一 个 或 多 个 SQL 语句 组 成 。 在 这 个 单元 中 , 每 个 MySQL 语句 是 相互 
依赖 的 。 而 整个 单独 单元 作为 一 个 不 可 分 隔 的 整体 ， 一 旦 单元 中 某 条 SQL 语句 执行 失败 或 产生 错误 ， 整 个 
单元 将 会 回 滚 ， 所 有 受到 影响 的 数据 将 返回 到 事务 开始 以 前 的 状态 ， 如 果 单 元 中 的 所 有 SQL 语句 均 执 行 成 
功 ， 则 事务 被 顺利 执行 。 

在 现实 生活 中 ， 事 务 处 理 数据 的 应 用 非常 广泛 ， 如 网 上 交易 、 银 行事 务 等 。 下 面 通过 网 上 交易 流程 向 
读者 展示 事务 的 概念 。 


相信 大 多 数 用 户 都 有 过 网 上 购物 的 体验 ， 即 用 户 登录 某 个 大 型 购 
物 网 站 ， 浏 览 该 网 站 中 所 陈列 的 商品 信息 ， 将 自己 喜欢 的 商品 放 入 购 
物 车 中 ， 选 购 完毕 后 ， 需 要 对 选 购 的 商品 进行 在 线 支付 ， 当 用 户 对 所 有 
选 商品 付款 完毕 后 ， 通 知 商家 发 货 。 在 此 过 程 中 ， 用 户 所 付 货款 并 未 
提交 到 商户 手中 ， 当 用 户 收 到 货物 之 后 ， 确 认 收 货 ， 商 家 才 收 到 商品 | 
货款 ， 整 个 交易 过 程 才 算 完 成 。 如 果 任何 一 步 操作 失败 ， 则 都 会 导致 和 
双方 陷入 尴 众 的 境界 ， 试 想 当 用 户 选 购 完 商品 并 付款 后 ， 在 发 货 过 程 一 站， 
中 取消 订单 。 这 时 商家 并 没有 得 到 取消 操作 提醒 ， 如 果 不 应 用 事务 处 rn 
理 , 则 在 用 户 取消 订单 操作 过 程 后, 商家 仍然 继续 将 用 户 所 订购 的 商品 一 一 一 
发 送 给 用 户 ， 这 会 导致 一 些 不 愉快 的 争端 。 故 在 整个 交易 过 程 中 ， 必 用 P 收 策 | 由 
须 采用 事务 来 对 网 上 交易 进行 回 滚 操作 。 其 流程 如 图 16.1 所 示 。 i | 
在 网 上 交易 流程 过 程 中 ， 商 家 与 用 户 的 交易 可 以 被 认为 是 一 个 事 ik 
务 处 理 过 程 。 其 中 整个 事务 过 程 中 ， 如 果 存 在 一 个 环节 失败 ， 都 可 能 二 
导致 双方 交易 的 失败 ， 如 前 面 事务 定义 所 说 ， 所 有 这 些 流程 都 应 该 被 


成 功 执行 。 在 MySQL 中 ， 如 果 其 中 有 任何 命令 失败 ， 都 会 导致 所 有 。 
操作 命令 被 撤销 ， 系 统 返回 未 操作 前 的 状态 ， 即 回 滚 到 初始 状态 。 添 图 161 应 用 事务 处 理由 上 交易 流程 
加 到 购物 车 、 在 线 付款 、 商 家 发 货 等 构成 了 一 个 基本 的 事务 。 整 个 交易 流程 可 以 被 看 做 一 个 完整 的 单元 ， 
用 于 实现 整体 事务 。 

通过 InnoDB 和 BDB 类 型 表 ，MySQL 事务 能 够 完全 满足 事务 安全 的 ACID 测试 。 但 是 并 不 是 所 有 表 类 
型 都 支持 事务 ， 如 MyISAM 类 型 表 就 不 能 支持 事务 。 只 能 通过 伪 事 务 对 表 实现 事务 处 理 。 


区 技巧 


16.1.1 原子 性 


ACID 指出 每 个 事务 型 RDBMS 必须 遵守 的 4 个 属性 ， 即 原子 性 、 一 致 性 、 孤 立 性 和 持久 性 。 


原子 性 意味 着 事物 的 整体 性 和 不 可 分 割 性 ， 类 似 于 化 学 中 的 原子 ， 事 务 就 具备 这 样 的 属性 ， 其 被 认为 

是 一 个 不 可 分 割 的 单元 。 假 设 一 个 事务 由 多 种 任务 组 成 ， 其 中 的 语句 必须 同时 操作 成 功 ， 才 可 以 认为 事务 

是 成 功 的 ， 否 则 将 回 深 到 初始 状态 。 从 网 上 交易 的 例子 可 以 看 出 ， 所 有 操作 的 成 功 是 保证 交易 完成 的 前 提 

条 件 ， 当 任何 一 个 环节 出 现 问 题 ， 即 导致 事务 回 深 ， 不 能 正常 完成 交易 过 程 。 即 所 有 事务 共同 进退 ， 保 证 
了 事务 的 整体 性 ， 这 就 是 事务 的 原子 性 。 
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原子 的 执行 是 一 个 全 部 发 生 或 全 部 失败 的 整体 过 程 。 在 一 个 原子 操作 中 ， 如 果 事 务 中 的 任何 一 个 语句 
失败 ， 前 面 执行 的 语句 都 将 被 返回 ， 以 保证 数据 的 整体 性 不 被 破坏 。 这 在 常用 的 系统 应 用 中 ， 为 保证 数据 
的 安全 性 起 到 一 定 作 用 。 


16.1.2 一 致 性 


在 MySQL 事务 处 理 过 程 中 ， 无 论 事务 是 完全 成 功 还 是 在 中 途 因 某 些 环节 失败 而 导致 失败 ， 事 务 使 系 
统 处 于 一 致 的 状态 时 ， 其 必须 保证 一 致 性 。 如 在 网 上 进行 转账 操作 的 过 程 中 ， 用 户 A 向 用 户 B 的 账户 中 转 
入 5000 元 ， 但 用 户 B 在 查询 转账 信息 时 ， 发 现 自己 的 账户 中 只 增加 了 3000 元 ， 这 样 不 能 使 整个 事务 达到 
一 致 性 。 例 如 ， 从 上 述 网 上 交易 的 例子 考虑 ， 当 交易 完成 后 ， 商 家 的 货物 会 减少 ， 不 可 能 出 现 当 商家 给 用 
户 发 货 后 货物 数量 不 变 ， 而 用 户 收 到 货物 后 不 增加 这 种 情况 。 

在 MySQL 中 ， 一 致 性 主要 由 MySQL 的 日 志 机 制 处 理 ， 它 记录 数据 库 的 所 有 变化 ， 为 事务 回复 提供 跟 
踪 记 录 。 如 果 系 统 在 事务 处 理 中 间 发 生 错误 ，MySQL 恢复 过 程 将 使 用 这 些 日 志 发 现 事务 是 否 已 经 完全 成 功 
执行 或 需要 返回 。 一 致 性 属性 保证 数据 库 从 不 返回 一 个 未 处 理 的 事务 。 


16.1.3 ”孤立 性 


孤立 性 是 指 每 个 事务 在 自己 的 空间 发 生 ， 和 其 他 发 生 在 系统 中 的 事务 隔离 ， 而 且 事务 的 结果 只 在 它 完 
全 被 执行 时 才能 看 到 。 即 使 系统 中 同时 发 生 多 个 事务 ， 孤 立 性 也 可 以 保证 特定 的 事务 在 完成 之 前 ， 其 结果 
是 不 被 公布 的 。 

当 系 统 支 持 多 个 同时 存在 的 用 户 和 连接 时 ， 系 统 必须 遵守 孤立 性 原则 ， 否 则 在 执行 过 程 中 可 能 导致 大 
量 数据 被 破坏 。 孤 立 性 保证 每 个 事务 完整 的 、 在 其 各 自 的 空间 内 被 顺利 执行 ， 保 证 事务 与 事务 之 间 不 会 相 
互 冲 突 。 


16.1.4 持久 性 


在 MySQL 中 ， 即 便 是 数据 库 系 统 崩 溃 ， 一 个 提交 的 事务 仍然 在 坚持 。 当 一 个 事务 完成 ， 数 据 库 的 日 志 
已 经 被 更 新 时 ， 持 久 性 即 可 发 挥 其 特有 功效 。 在 MySQL 中 ， 如 果 系统 崩溃 或 者 数据 存储 介质 被 破坏 ， 通 过 
使 用 日 志 ， 系 统 能 够 恢复 在 重启 前 进行 的 最 后 一 次 成 功 更 新 ， 可 以 反映 系统 崩溃 时 处 于 执行 过 程 的 事务 的 
变化 。 

MySQL 的 持久 性 是 通过 一 条 记录 事务 过 程 中 系统 变化 的 二 进 制 事务 日 志文 件 来 实现 的 。 如 果 遇 到 硬件 
损坏 或 者 系统 异常 关机 ， 系 统 在 下 一 次 启动 时 ， 通 过 使 用 最 后 的 备份 和 日 志 就 可 以 恢复 丢失 的 数据 。 


了 二 默认 情况 下 ，InnoDB 表 持久 性 最 久 ，MyISAM 表 具 有 部 分 持久 性 . 


16.2 MySQL 事务 的 创建 与 存在 周期 


吉 4 视频 讲解 : 光盘 \TIM\Video\ 第 16 章 \MySQL 事务 的 创建 与 存在 周期 .exe 
通过 上 述 对 事务 定义 的 叙述 和 事务 特性 的 讲解 ， 相 信 读 者 已 经 对 事务 有 一 个 初步 的 认识 。 下 面 将 对 事 
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务 的 存在 周期 做 详细 讲解 。 首 先 ， 向 读者 介绍 如 何在 MySQL 中 创建 事务 。 

创建 事务 的 一 般 过 程 是 初始 化 事务 、 创 建 事务 、 应 用 SELECT 语句 查询 数据 是 否 被 录入 和 提交 事务 。 
如 果 用 户 不 在 操作 完 数据 库 后 执行 事务 提交 ， 则 系统 会 默认 执行 回 滚 操作 。 如 果 用 户 在 提交 事务 前 选择 撤 
销 事务 ， 则 用 户 在 撤销 前 的 所 有 事务 将 被 取消 。 数 据 库 系 统 会 回 到 初始 状态 。 


和 ,注意 默认 情况 下 , 在 MySQL 中 创建 的 数据 表 类 型 都 是 MyISAM, 但 是 该 类 型 的 数据 表 并 不 能 支持 
事务 。 所 以 , 如 果 想 让 数据 表 支 持 事务 处 理 能 力 ,必须 将 当前 操作 数据 表 的 类 型 设置 为 noDB 或 BDB。 


在 创建 事务 的 过 程 中 ， 需 要 创建 一 个 ImnoDB 或 BDB 类 型 的 数据 表 。 其 基本 命令 结构 如 下 : 

CREATE TABLE table_name(field_defintions) TYPE = INNODB/BDB; 

其 中 ，table_name 为 表 名 ; field_defintions 为 表 内 定义 的 字段 等 属性 ，TYPE 可 指定 数据 表 的 类 型 ， 既 
可 以 是 InnoDB 类 型 ， 可 以 是 BDB 类 型 。 

如 果 希 望 让 已 经 存在 的 表 支 持 事务 处 理 ， 则 可 以 应 用 ALTER TABLE 命令 指定 数据 表 的 类 型 ， 即 可 实 
现 对 表 类 型 的 更 改 操作 ， 使 原本 不 支持 事务 的 数据 表 更 改 为 支持 事务 处 理 的 类 型 。 其 命令 如 下 : 

ALTER TABLE table_name TYPE= INNODB/BDB; 

更 改 完 表 的 类 型 后 ， 即 可 使 数据 表 支 持 事务 处 理 。 
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史 技巧 应 用 ALTER TABLE 操作 可 能 会 导致 数据 库 中 的 数据 丢失 ， 因 此 为 了 避免 非 预 期 结果 出 现 ， 
在 使 用 ALTER TABLE 命令 前 ， 需 要 创建 一 个 表 备 份 。 


16.2.1 初始 化 事务 


初始 化 MySQL 事务 ， 首 先 声 明 初 始 化 MySQL 事务 后 所 有 的 SQL 语句 为 一 个 单元 。 在 MySQL 中 , 应 
用 START TRANSACTION 命令 来 标记 一 个 事务 的 开始 。 初 始 化 事务 的 结构 如 下 : 

START TRANSACTION; 

另外 ， 也 可 以 使 用 BEGIN 或 者 BEGIN WORK 命令 初始 化 事务 ,通常 START TRANSACTION 命令 后 
面 跟随 的 是 组 成 事务 的 SQL 语句 。 

在 命令 提示 符 中 输入 如 下 命令 : 

start transaction; 

如 果 输 入 以 上 代码 后 ，MySQL 数据 库 没 有 给 出 警告 提示 或 返回 错误 信息 ， 则 说 明 事务 初始 化 成 功 ， 可 
以 继续 执行 下 一 步 操作 。 


16.2.2 ”创建 事务 


例 16.1 初始 化 事务 成 功 后 ， 可 以 创建 事务 。 这 里 以 向 名 称 为 connection 的 数据 表 中 插入 一 条 记录 为 
例 ， 讲 解 事务 的 创建 。 首 先 打开 数据 库 ， 选 定 某 个 数据 库 ， 然 后 初始 化 事务 ， 最 后 创建 事务 ， 向 指定 的 数 
据 表 中 添加 记录 。《〈 实 例 位置 ， 光盘 \TMNInstance\16\16.1) 

其 代码 如 下 : 

start transaction; 

insert into connection(email,cellphone,QQ,sid) 

values('mrsoft@mrsoft.com',7856456789,92343432,3); 
其 运行 结果 如 图 16.2 所 示 。 
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16.2.3 应 用 SELECT 语句 查看 数据 是 否 被 正确 输入 


事务 创建 成 功 后 ， 建 议 读者 通过 SELECT 语句 查看 数据 是 否 被 正确 输入 。 

例 16.2 ”在 事务 初始 化 成 功 后 创建 事务 。 (实例 位 置 : 光盘 \TM\Instance\16\16.2) 
在 命令 提示 符 中 输入 如 下 指令 : 

SELECT* FROM connection WHERE sid=3; 

其 运行 结果 如 图 16.3 所 示 。 


有 图 16.2 创建 事务 图 16.3 查看 数据 是 否 被 正确 输入 


[3 技巧 在 插入 新 表 为 IhnoDB 类 型 或 更 改 原来 表 类 型 为 InnoDB 时 ， 如 果 在 输入 命令 提示 后 ，MySQL 
提示 “The 'InnoDB' feature is disabled; you need InnoDB 'to have it working” 警 告 ， 则 说 明 InnoDB 表 类 型 
并 没有 被 开启 ， 用 户 需 要 找到 MySQL 文件 目录 下 的 my.ini 文 件 ， 定 位 skip innodb 选项 位 置 ， 将 原来 的 
skip”innodb 改 为 #skip_innodb 后 保存 该 文件 ， 重 新 启动 MYSQL 服务 器 ， 即 可 令 数据 库 支 持 InnoDB 类 


16.2.4 ”提交 事务 


在 用 户 没 有 提交 事务 之 前 ， 当 其 他 用 户 连 接 MySQL 服务 器 时 , 应 用 SELECT 语句 查询 结果 ， 则 不 会 显 
示 没 有 提交 的 事务 。 当 且 仅 当 用 户 成 功 提交 事务 后 ， 其 他 用 户 才 可 以 通过 SELECT 语句 查询 事务 结果 。 由 
事务 的 特性 可 知 ， 事 务 具 有 孤立 性 ， 当 事务 处 在 处 理 过程 中 时 ， 其 实 MySQL 并 未 将 结果 写 入 磁盘 中 ， 这 样 
-来 ， 这 些 正 在 处 理 的 事务 相对 其 他 用 户 是 不 可 见 的 。 一 旦 数据 被 正确 插入 ， 用 户 可 以 使 用 COMMIT 命令 
提交 事务 。 提 交 事 务 的 命令 结构 如 下 : 
COMMIT 


- 旦 当前 执行 事务 的 用 户 提交 当前 事务 ， 则 其 他 用 户 就 可 以 通过 会 话 查询 结果 。 
16.2.5 ”撤销 事务 〈 事 务 回 滚 ) 


撤销 事务 ， 又 被 称 做 事务 回 滚 ， 即 事务 被 开启 、 用 户 输入 的 SQL 语句 被 执行 后 ， 如 果 用 户 想 要 撤销 刚 
才 的 数据 库 操作 ， 可 使 用 ROLLBACK 命令 撤销 数据 库 中 的 所 有 变化 。ROLLBACK 命令 的 结构 如 下 : 
ROLLBACK 

输入 回 滚 操作 后 ， 可 以 通过 SELECT 语句 查看 是 否 执 行 了 
看 16.2.2 节 中 插入 的 数据 是 否 存 在 ， 其 运行 结果 如 图 16.4 所 示 。 


回 


滚 操作 。 例 如 ， 可 以 通过 SELECT 语句 查 


人 注意 如 果 执行 一 个 回 滚 操作 ， 则 在 START TRANSACTION 命令 后 的 所 有 SQL 语句 都 将 执行 回 滚 


操作 。 故 在 执行 事务 回 滚 前 ， 用 户 需 要 慎重 选择 是 否 执行 回 滚 操作 。 如 果 开 局 事务 后 ， 没 有 提交 事务 ， 
则 事务 默认 为 自动 回 滚 状态 ， 即 不 保存 用 户 之 前 的 任何 操作 .。 
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| se 
账 方式 交易 ， 用户 A 将 个 人 账户 的 部 分 存款 转移 到 用 户 B 的 个 人 账户 过 程 中 , 若 银 行 的 数据 库 系 统 突然 
发 生 错 误 或 异常 ， 则 交易 事务 提交 失败 ， 系 统 执 行 回 滚 操作 ， 恢 复 到 交易 的 初始 状态 。 这 样 可 以 避免 因 
特殊 情况 而 导致 事务 提交 失败 ， 从 而 减少 不 必要 的 损失 。 


16.2.6 事务 的 存在 周期 


事务 的 周期 由 用 户 在 命令 提示 符 中 输入 START TRANSACTION 指令 开始 ， 直 至 用 户 输入 COMMIT 结 
束 ， 如 图 16.5 所 示 为 一 个 简单 事务 存在 周期 流程 图 。 


( 初 如 化 数据 库 状态 )—on 


于 始 事 务 


选择 、 插 入 、 更 新 …… 
提交 回流 
a S 
| +, 一 
/ 
16.4 执行 回 滚 操作 后 应 用 SELECT 查询 图 16.5 事务 的 存在 周期 


[a 技巧 事务 不 支持 嵌 套 功能 , 当 第 一 个 事务 未 结束 又 重新 打开 一 个 事务 时 , 则 前 一 个 事务 会 自动 提交 。 
同样 ，MySQL 命令 中 很 多 命令 都 会 隐藏 执行 COMMIT 命令 。 


16.3 MySQL 行为 


在 MySQL 中 ， 存 在 两 个 可 以 控制 行为 的 变量 ， 分 别 是 AUTOCOMMIT 变量 和 TRANSACTION 
ISOLACTION LEVEL 变量 。 


16.3.1 自动 提交 


在 MySQL 中 ， 如 果 不 更 改 其 自动 提交 变量 ， 系 统 会 自动 向 数据 库 提交 结果 。 用 户 在 执行 数据 库 操作 过 
程 中 ， 不 需要 使 用 START TRANSACTION 语句 开始 事务 ， 应 用 COMMIT 或 者 ROLLBACK 提交 事务 或 执 
行 回 滚 操作 。 如 果 用 户 希望 通过 控制 MySQL 自动 提交 参数 ， 可 以 更 改 提交 模式 ， 这 一 更 改过 程 是 通过 设置 
AUTOCOMMIT 变量 来 实现 的 。 

下 面 通过 一 个 实例 向 读者 展示 如 何 关闭 自动 提交 功能 。 在 命令 提示 符 中 输入 以 下 命令 : 
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SET AUTOCOMMIT=0 ; 

此 时 即 可 关闭 自动 提交 功能 。 只 有 当 用 户 输入 COMMIT 命令 后 ，MySQL 才 将 数据 表 中 的 资料 提交 到 
数据 库 中 ， 如 果 不 提 交 事 务 ， 而 终止 MySQL 会 话 ， 数 据 库 将 会 自动 执行 回 滚 操作 。 

例 16.3 在 关闭 自动 提交 功能 后 ， 查 询 数据 表 中 数据 ， 并 且 向 数据 表 中 添加 一 条 记录 。 (实例 位 置 : 
光盘 \TM\Instance\16\16.3) 

其 命令 如 下 : 

set autocommit=0; 


select * from timeinfo; 
insert into timeinfo(info) values('test autocommit); 


以 上 代码 的 运行 结果 如 图 16.6 所 示 。 
由 于 用 户 已 经 关闭 自动 提交 功能 ， 所 以 图 16.6 中 所 示 的 添加 操作 因 没有 执行 事务 的 提交 操作 ， 而 导致 
数据 没有 成 功 添加 。 再 次 查询 数据 表 中 的 数据 ， 运 行 结果 如 图 16.7 所 示 。 


图 16.6 取消 自动 提交 操作 图 16.7 应 用 SELECT 查询 自动 提交 关闭 后 的 数据 


结果 表明 ， 之 前 插入 的 数据 并 未 插入 到 数据 库 中 。 另 外 ， 我 们 可 以 通过 
查看 “@@AUTOCOMMIT” 变 量 来 查看 当前 自动 提交 状态 ， 查 看 此 变量 同 
样 应 用 SELECT 语句 ， 其 运行 结果 如 图 16.8 所 示 。 


16.3.2 事务 的 孤立 级 图 16.8 查看 自动 提交 变量 


事务 具有 独立 的 空间 ， 在 MySQL 服务 器 中 ， 用 户 通过 不 同 的 会 话 执行 不 同 的 事务 ， 在 多 用 户 环境 中 ， 
许多 RDBMS 会 话 在 任意 指定 时 刻 都 是 活动 的 。 为 了 使 这 些 事务 互 不 影响 ， 保 证 数据 库 性 能 不 受到 影响 ， 
采用 事务 的 孤立 级 是 十 分 有 必要 的 。 
孤立 级 在 整个 事务 中 起 到 了 很 重要 的 作用 ， 如 果 没 有 事务 的 孤立 性 ， 不 同 的 SELECT 语句 将 会 在 同一 
事务 的 环境 中 检索 到 不 同 的 结果 ， 这 将 导致 数据 的 不 一 致 性 ， 给 不 同 的 用 户 造 成 困扰 ， 这 样 一 来 ， 用 户 就 
不 能 以 查询 的 结果 集 作为 计算 基础 。 所 以 孤立 性 强制 保持 每 个 事务 的 独立 性 ， 以 此 来 保证 事务 中 看 到 一 致 
的 数据 。 
基于 ANSIISO SQL 规范 ，MySQL 提供 了 以 下 4 种 孤立 级 。 
SERIALIZABLE〈 序 列 化 ) : 顾名思义 ， 以 序列 的 形式 对 事务 进行 处 理 ， 该 孤立 级 的 特点 是 只 有 
当 事 务 提 交 后 ， 用 户 才能 从 数据 库 中 查看 数据 的 变化 。 该 孤立 级 运行 会 影响 MySQL 的 性 能 ， 因 为 
需要 占用 大 量 资 源 ， 以 保证 使 大 量 事务 在 任意 时 间 不 被 用 户 看 到 。 
REPEATABLE READ (可 重读 ) : 对 于 应 用 程序 的 安全 性 做 出 部 分 妥协 ， 以 提高 其 性 能 。 事 务 在 
该 孤立 级 上 不 会 被 看 成 一 个 序列 ， 不 过 当前 在 执行 事务 的 过 程 中 ， 用 户 仍然 看 不 到 事务 的 过 程 。 
直到 事务 提交 为 止 ， 用 户 才能 够 看 到 事务 的 变化 结果 。 
READ COMMITTED (提交 后 读 ) : 提交 后 读 孤 立 级 的 安全 性 比重 复读 安全 性 要 低 。 在 这 一 级 的 
事务 ， 用 户 可 以 看 到 其 他 事务 添加 的 新 记录 。 在 事务 处 理 时 ， 如 果 存 在 其 他 用 户 同 时 对 事务 的 相 
应 表 进 行 修改 ， 那 么 在 同一 事务 中 不 同时 间 内 ， 应 用 SELECT 语句 可 能 返回 不 同 的 结果 集 。 
READ UNCOMMITTED (未 提交 读 ) : 该 孤立 级 提供 事务 之 间 的 最 小 程度 间隔 ， 容 易 产生 虚幻 读 
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操作 。 其 他 用 户 可 以 在 该 孤立 级 上 看 到 未 提交 的 事务 。 


16.3.3 ”修改 事务 的 孤立 级 


在 MySQL 中 ,可 以 使 用 TRANSACTION ISOLATION LEVEL 变量 修改 事务 孤立 级 ,其 中 ,MySQL 的 
默认 孤立 级 为 REPEATABLE READ (可 重读 ), 用 户 可 以 使 用 SELECT 命令 获取 当前 事务 孤立 级 变量 的 值 ， 
其 命令 如 下 : 

SELECT @@tx_isolation; 

例 16.4 ”如 果 希 望 修改 事务 的 孤立 级 ， 可 以 通过 SET 命令 设置 其 不 
同 值 ， 来 修改 事务 的 孤立 级 ， 操 作 命 令 如 图 16.9 所 示 。 (实例 位 置 : 光盘 \ ES 
TM\Instance\16\16.4) 

图 16.9 设置 事务 孤立 级 


[Ea 如 果 想 修改 事物 的 孤立 级 ， 必 须 首先 获取 SUPER 优先 权 ， 以 便 可 以 顺利 执行 修改 操作 。 


16.4 事务 和 性 能 


从 16.3 节 中 可 以 看 出 ， 应 用 不 同 孤立 级 的 事务 可 能 会 对 系统 造成 一 系列 影响 ， 采 用 不 同 孤 立 级 处 理事 
务 ， 可 能 会 对 系统 稳定 性 和 安全 性 等 诸多 因素 造成 影响 。 另 外 ， 有 些 数 据 库 操作 中 ， 不 需要 应 用 事务 处 理 ， 
则 用 户 需 要 选择 合适 的 数据 表 类 型 。 在 选择 表 类 型 时 ， 应 该 考虑 数据 表 具 有 完善 的 功能 ， 且 高 效 执行 时 不 
会 对 系统 增加 额外 的 负担 。 


16.4.1 应 用 小 事务 


应 用 小 事务 的 意义 在 于 :保证 每 个 事务 不 会 在 执行 前 等 待 很 长 时 间 ， 从 而 避免 因为 各 个 事务 互相 等 待 
而 导致 系统 的 性 能 大 幅度 下 降 。 用 户 在 应 用 少数 大 事务 时 ， 可 
能 无 法 看 出 因 事 务 间 互 相等 待 而 导致 系统 性 能 下 降 ， 但 是 当 系 yw j 
统 中 存在 处 理 量 很 大 的 数据 库 或 多 种 复杂 事务 时 ， 因 长 时 间 等 / 


待 而 导致 系统 性 能 下 降 就 很 明显 。 所 以 ， 应 用 小 事务 可 以 保证 / 
系统 的 性 能 ， 小 事务 可 以 快速 变化 或 退出 ， 这 样 ， 其 他 在 队列 qz 十 / 
中 准备 就 绪 的 事务 就 不 会 受到 明显 影响 。 
提交 后 读 十 / 

16.4.2 ”选择 合适 的 孤立 级 

事务 的 性 能 与 其 对 服务 器 产生 的 负载 成 反比 ， 即 当 事 务 孤 。 Sa 十 一 一 一 人 mm 
立 级 越 高 ， 其 性 能 越 低 ， 但 是 其 安全 性 也 越 高 。 事 务 性 能 和 孤 和 
立 级 关系 如 图 16.10 所 示 。 所 以 只 有 选择 适当 的 孤立 级 ， 才 能 
有 效 地 提高 MySQL 系统 性 能 和 应 用 性 。 图 16.10 事务 孤立 级 和 性 能 的 关系 


从 图 16.10 可 以 看 出 ,虽然 稳定 性 随 着 孤立 级 的 增高 而 改变 ,但 是 并 不 代表 孤立 级 的 稳定 性 越 高 ， 其 灵 
活性 越 高 ， 故 用 户 在 选择 孤立 级 时 ， 需 要 根据 实际 情况 选择 适合 应 用 的 孤立 级 ， 切 勿 生 搬 硬 套 。 
@ 
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16.4.3 ” 死 锁 的 概念 与 避免 


死 锁 ， 即 当 两 个 或 者 多 个 处 于 不 同 序列 的 用 户 打算 同时 更 新 某 相同 的 数据 库 时 ， 因 互相 等 待 对 方 释放 
权限 而 导致 双方 一 直 处 于 等 待 状态 。 在 实际 应 用 中 ， 两 个 不 同 序列 的 客户 打算 同时 对 数据 执行 操作 ， 极 有 
可 能 产生 死 锁 。 更 具体 地 讲 ， 当 两 个 事务 相互 等 待 操作 对 方 释放 所 持 有 的 资源 ， 而 导致 两 个 事务 都 无 法 操 
作对 方 持 有 的 资源 ， 这 样 无 限期 的 等 待 被 称 做 死 锁 。 

不 过 ，MySQL 的 InnoDB 表 处 理 程序 具有 检查 死 锁 功 能 ， 如 果 该 处 理 程序 发 现 用 户 在 操作 过 程 中 产生 
死 锁 ， 将 立刻 通过 撤销 操作 来 撤销 其 中 一 个 事务 ， 以 便 使 死 锁 消失 。 这 样 就 可 以 使 另 一 个 事务 获取 对 方 所 
占有 的 资源 而 执行 逻辑 操作 。 


16.5 MySQL 伪 事 务 


在 MySQL 中 ，InnoDB 和 BDB 类 型 表 可 以 支持 事务 处 理 ， 但 是 MyISAM 类 型 表 并 不 能 支持 事务 处 理 ， 
对 于 该 类 型 表 ， 用 户 可 以 选择 应 用 表 锁 定 来 蔡 代 事务 。 这 种 引用 表 锁 定 来 蔡 代 事务 的 事件 被 称 做 伪 事 务 。 
使 用 表 锁 来 锁定 表 的 操作 ， 可 以 加 强 非 事务 表 在 执行 过 程 的 安全 性 和 稳定 性 。 


16.5.1 用 表 锁 定 代 替 事 务 


在 MySQL 的 MyISAM 类 型 数据 表 中 ， 并 不 支持 COMMIT (提交 ) 和 ROLLBACK ( 回 深 ) 命令 。 当 
用 户 对 数据 库 执行 插 入 、 删 除 、 更 新 等 操作 时 ， 这 些 变化 的 数据 都 被 立刻 保存 在 磁盘 中 。 这 样 ， 在 多 用 户 
环境 中 会 导致 诸多 问题 ， 为 了 避免 同一 时 间 有 多 个 用 户 对 数据 库 中 指定 表 进行 操作 ， 可 以 应 用 表 锁 定 来 避 
免 在 用 户 操作 数据 表 过 程 中 受到 干扰 。 当 且 仅 当 该 用 户 释放 表 的 操作 锁定 后 ， 其 他 用 户 才 可 以 访问 这 些 修 
改 后 的 数据 表 。 

设置 表 锁定 代替 事务 的 基本 步骤 如 下 。 

(1) 为 指定 数据 表 添加 锁定 。 其 语法 如 下 : 

LOCK TABLES table_name lock_type,... 

其 中 ，table_name 为 被 锁定 的 表 名 ; lock_type 为 锁定 类 型 ， 该 类 型 包括 以 读 方 式 (READ) 锁定 表 、 以 
写 方式 (WRITE) 锁定 表 。 

(2) 执行 数据 表 的 操作 ， 可 以 添加 、 删 除 或 者 更 改 部 分 数据 。 

(3) 完成 对 锁定 数据 表 的 操作 后 ， 需 要 对 该 表 进行 解锁 操作 ， 释 放 该 表 的 锁定 状态 。 其 语法 如 下 : 

UNLOCK TABLES 

下 面 通过 实例 向 读者 展示 如 何以 读 方式 和 以 写 方 式 锁定 数据 表 。 

1. 以 读 方 式 锁定 数据 表 


以 读 方式 锁定 数据 表 是 设置 锁定 用 户 的 其 他 方式 操作 ， 如 删除 、 插 入 、 更 新 都 不 被 允许 ， 直 至 用 户 进 
行 解锁 操作 。 
例 16.5 锁定 studentinfo 数据 表 。 


首先 ， 在 命令 提示 符 下 输入 如 下 代码 : 〔 实 例 位 置 ， 光盘 \TM\Instance\16\16.5) 
lock table studentinfo read; 


@ 
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然后 , 应 用 SELECT 语句 查看 表 中 的 信息 , 其 运行 结果 
如 图 16.11 所 示 。 

下 面 ， 尝 试 向 该 数据 表 中 插入 一 条 数据 ， 其 运行 结果 如 
图 16.12 所 示 。 

从 上 述 结果 可 以 看 出 ， 当 用 户 试 图 向 数据 库 插入 数据 


时 ， 将 会 返回 失败 信息 。 当 将 锁定 的 表 解 锁 后 ， 再 次 执行 插 es 
入 操作 ， 其 运行 结果 如 图 16.13 所 示 。 图 16.11 查看 以 读 方 式 锁定 的 studentinfo 表 


图 16.12 向 以 读 方 式 锁定 的 表 中 插入 数据 图 16.13 ”向 解锁 后 的 数据 表 中 添加 数据 
锁定 被 释放 后 ， 用 户 可 以 对 数据 库 执行 添加 、 删 除 、 更 新 等 操作 。 


-和 


[a 技巧 其 中 ,lock type 参数 中 ,用 户 指定 数据 表 以 读 方式 (READ ) 锁 定数 据 表 的 变 体 为 READ LOCAL 
锁定 。 其 与 READ 锁定 的 不 同 点 是 ， 该 参数 所 指定 的 用 户 会 话 可 以 执行 INSERT 操作 。 它 是 为 了 使 用 
MySQL dump 工具 而 创建 的 一 种 变 体形 式 。 


2. 以 写 方 式 锁定 数据 表 

与 读 方式 锁定 表 类 似 ， 表 
其 他 用 户 不 能 进行 任何 读 

lock table studentinfo wri 

例 16.6 ”因为 该 表 为 写 锁定 ， 用 户 可 以 对 数据 库 的 数据 执行 修改 、 添 加 、 删 除 等 操作 。 在 该 命令 提示 
符 中 应 用 SELECT 语句 查询 该 锁定 表 , 其 运行 结果 如 图 16.14 所 示 。( 实 例 位置 : 光盘 \TM\Instance\16\16.6) 

从 图 16.14 中 可 以 看 到 ,当前 用 户 应 用 SELECT 语句 仍然 可 以 查询 该 表 的 数据 , 并 没有 限制 用 户 对 数据 
表 的 读 操 作 。 这 是 因为 ， 以 写 方式 锁定 数据 表 并 不 能 限制 当前 锁定 用 户 的 查询 操作 。 下 面 打 开 一 个 新 用 户 
会 话 ， 即 保持 图 16.14 所 示 窗 口 不 被 关闭 ， 重 新 打开 一 个 新 的 MySQL 连接 ， 并 执行 上 述 过 程 。 其 运行 结果 
如 图 16.15 所 示 。 


写 锁定 是 设置 用 户 可 以 修改 数据 表 中 的 数据 ， 但 是 除 自 己 以 外 ， 会 话 中 的 
在 命令 提示 符 中 输入 如 下 命令 : 


图 16.14 查询 应 用 写 操作 锁定 的 studentinfo 数据 表 图 16.15 打开 新 会 话 查询 被 锁定 的 数据 表 


在 新 打开 的 命令 提示 界面 可 以 看 到 ， 应 用 SELECT 语句 执行 查询 操作 ， 并 没有 结果 显示 ， 这 是 因为 之 
前 该 表 以 写 方式 锁定 。 故 当 操作 用 户 释放 该 数据 表 锁 定 后 ， 其 他 用 户 才 可 以 通过 SELECT 语句 查看 之 前 被 
锁定 的 数据 表 。 在 命令 提示 符 中 输入 如 下 代码 : 

UNLOCK TABLES; 

这 时 ， 在 第 二 次 打开 的 命令 提示 符 中 ， 即 可 出 现 如 图 16.14 所 示 的 结果 。 当 数据 表 被 释放 锁定 后 ， 其 他 
访问 数据 库 的 用 户 即 可 查看 数据 表 的 内 容 。 
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BE 使 用 UNLOCK TABLE 命令 后 ， 将 会 释放 所 有 当前 处 于 锁定 状态 的 数据 表 。 


16.5.2 ”应 用 表 锁 实现 伪 事 务 


相信 读者 通过 上 面 的 学 习 已 经 了 解 什么 是 表 锁 ， 下 面 通过 使 用 表 锁 对 MyISAM 表 进 行 锁定 操作 ， 以 此 
过 程 来 代 蔡 事务 型 表 InnoDB， 即 应 用 表 锁 来 实现 伪 事 务 。 

实现 伪 事 务 的 一 般 步骤 如 下 。 

(1) 对 数据 库 中 的 数据 表 进 行 锁定 操作 ， 可 以 对 多 个 表 做 不 同 的 方式 锁定 。 其 代码 格式 如 下 : 

LOCK TABLE table_name1 lock_type1, table_name2 lock_type2,... 

(2) 执行 数据 库 操作 ， 向 锁定 的 数据 表 中 执行 添加 、 删 除 、 修 改 操 等 操作 。 如 前 面 提 到 的 INSERT、 
UPDATE、DELETE 等 操作 。 用 户 可 以 对 锁定 的 数据 表 执 行 上 述 操作 ， 在 执行 过 程 中 ， 该 伪 事 务 所 产生 的 结 


果 是 不 会 被 其 他 用 户 更 改 的 。 
(3) 释放 锁定 的 数据 表 ， 以 便 让 正在 队列 中 等 待 查看 或 操作 的 其 他 用 户 可 以 浏览 数据 表 中 的 数据 或 对 
操作 表 执 行 各 种 数据 的 操作 。 


如 果 存 在 其 他 会 话 要 求 访问 己 锁 定 的 多 个 表格 ， 则 该 会 话 必须 被 迫 等 待 当前 锁定 用 户 释放 锁定 表 ， 才 
允许 其 他 会 话 访问 该 数据 表 ， 表 锁定 使 不 同 会 话 执行 的 数据 库 操作 彼此 独立 。 应 用 数据 表 锁定 方式 可 以 使 
不 支持 事务 类 型 的 表 实现 伪 事 务 。 


16.6 实 战 


16.6.1 使 用 事务 处 理 技术 实现 银行 的 安全 转账 (PHP) 


例 16.7 在 实现 银行 转账 过 程 中 , 发 生意 外 是 在 所 难免 的 ， 为 模拟 转账 
了 避免 因 意外 而 造成 不 必要 的 损失 ,在 银行 转账 时 经 常 使 用 事务 处 
理 方式 。 运 行 本 实例 ， 效 果 如 图 16.16 所 示 ， 在 文本 框 中 输入 要 转 Pm | 
给 B 账户 的 金额 后 , 单 击 “ 转 账 ”按钮 即 可 实现 转账 。( 实 例 位置 : 
光盘 \TM\Instance\16\16.7) Ea 
实现 过 程 如 下 所 示 。 
(1) 利用 mysqli 扩展 技术 实现 与 MySQL 数据 库 的 连接 ， 代 16.16 银行 转账 
码 如 下 : 
<?php 
$conn=new mysqli("localhost","root","111","db_database16"); /| 连接 数据 库 
S$conn->query("set names utf8"); /设置 编码 格式 
人 
(2) 查询 A 账户 与 B 账户 的 现 有 金额 并 显示 在 页 面 中 。 代 码 如 下 : 
include_once("conn.php"); // 包 含 数据 库 连 接 文件 conn.php 
$sql=$conn->query("select * from tb_zy where flag='mrsoft"); /查询 mrsoft 用 户 的 信息 
Sinfo=$sql->fetch_array(MYSQLI_ASSOC): 
$sql1=$conn->query("select * from tb_zy where fag= mr"); /查询 mr 用 户 的 信息 


Sinfo1=$sql1->fetch_array(MYSQLI_ASSOC); 


他 
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(3) 实现 转账 ， 其 主要 代码 如 下 : 


S$tob=$_POSTT[tob]; /接收 提交 的 转账 金额 

include_once("conn.php"); // 包 含 数 据 库 连 接 文件 conn.php 

$conn->autocommit(false); /取消 查询 自动 提交 

if(!$conn->query("update tb_zy set money=money-".$tob." where flag='mrsoft")){ /减少 指定 账户 所 要 转账 的 金额 
$conn->rollback(); // 如 果 失 败 则 事务 回 滚 

a tb_zy set money=money+".$tob." where flag='mr")){ /增加 指定 账户 所 要 的 转账 金额 
$conn->rollback(); // 如 果 失 败 则 事务 回 滚 

小 

$conn->commit(); /提交 查询 

S$conn->autocommit(true); /恢复 查询 自动 提交 

echo "<script>window.location.href='"index.php';</script>"; /| 重 定向 到 首页 


16.6.2” 批 处 理 中 使 用 事务 (Java) 


例 16.8 ”在 企业 级 的 应 用 程序 中 经 常会 遇 到 对 多 个 数据 表 同时 存 取 的 情况 ， 最 明显 的 例子 是 银行 的 转 
账 业 务 ， 从 汇款 账户 中 减 去 指定 金额 ， 并 将 该 金额 添加 至 收 款 账户 中 ， 但 如 果 在 转账 的 过 程 中 发 生 程序 错 
误 或 者 系统 断 电 等 意外 情况 ， 就 可 能 导致 汇款 账户 的 余额 已 经 减少 而 收 款 账户 的 余额 还 没有 增加 ， 这 就 需 
要 应 用 事务 对 该 问题 进行 处 理 。 本 实例 模拟 银行 转账 系统 ， 通 过 事务 保证 转账 业务 的 顺利 进行 。( 实 例 位 
置 : 光盘 \TM\Instance\16\16.8) 

实现 过 程 如 下 所 示 。 

(1) 在 项 目 中 创建 BatchAffair 类 , 在 该 类 中 定义 操作 数据 的 各 种 方法 ,其 中 定义 获取 账户 表 tb_transition 
中 所 有 账户 方法 selectIds0， 该 方法 以 List 集合 对 象 作为 返回 值 。 具 体 代码 如 下 : 

public List selectlds() { 


conn = getConn(); // 获 取 数 据 库 连接 
Statement cs = null; // 定 义 CallableStatement 对 象 
String sql = "Select accoutNumber from tb_transition"; /定义 查询 视图 的 SQL 语句 
List list = new ArrayList(); /定义 保存 查询 结果 的 List 集合 
try{ 
cs = conn.createStatement(); /实例 化 Statement 对 象 
ResultSet rest = cs.executeQuery(sql); /执行 SQL 语句 
while (rest.next()) { /| 循环 遍历 查询 结果 集 
String accoutNumber = rest.getString(1); 
list.add(accoutNumber); 
站 
} catch (SQLException e){ 
e.printStackTrace(); 
return list; 


} 
(2) 在 BatchAffair 类 中 定义 转账 方法 Batch0， 该 方法 包含 两 个 String 类 型 的 参数 与 一 个 float 类 型 的 
参数 ， 分 别 用 于 指定 转账 的 账户 、 转 入 的 账户 和 转账 的 金额 。 具 体 代码 如 下 : 
public void Batch(String incomeld, String gold, float money) throws SQLException { 


ty{ 
conn = getConn(); // 获 取 数 据 库 连 接 
boolean autoCommit = conn.getAutoCommit(); 
conn.setAutoCommit(false); 
Statement cs = null; /定义 Statement 对 象 


全 
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cs = conn.createStatement(); /实例 化 Statement 对 象 
cs.addBatch("update tb_transition set deposit = deposit-" + money 
+" ,transition = transition-" + money 
+ " Where accoutNumber = " + gold); /定义 修改 转账 表 中 的 数据 方法 
cs.addBatch("update tb_transition set deposit = deposit+" + money 
+" ,Shift = shift+" + money + " where accoutNumber =" 
+ incomeld); 
cs.executeBatch(); /批量 执行 SQL 语句 
cs.close(); // 将 Statement 对 象 关闭 
conn.commit(); 
conn.setAutoCommit(autoCommit); 
conn.close(); 
} catch (Exception e) { 
conn.rollback(); 
e.printStackTrace(); 


} 
运行 结果 如 图 16.17 所 示 。 
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图 16.17 银行 转账 业务 
16.7 小 结 


本 章 对 MySQL 中 事务 的 创建 、 提 交 、 撤 销 ， 以 及 事务 的 存在 周期 进行 了 详细 讲解 ， 并 通过 举例 说 明 ， 
使 读者 更 好 地 理解 所 学 知识 的 用 法 。 在 阅读 本 章 时 ， 读 者 应 该 重点 掌握 如 何 自动 提交 事务 和 修改 事务 的 孤 
立 级 。 同 时 ， 应 该 了 解 MySQL 事务 和 性 能 ， 以 及 MySQL 伪 事 务 。 


16.8 学习 成 果 检 验 


1. 如 何 初 始 化 事务 ? (实例 位 置 ， 光盘 \TM\Instance\16\16.9) 
2. MySQL 中 可 以 控制 行为 的 变量 为 哪 两 个 ? 《实例 位 置 : 光盘 \TM\Instance\16\16.10) 
3. 如 何 设置 表 锁 定 代 葵 事 务 ? (实例 位 置 : 光盘 \TMNInstance\16\16.11) 


@ 
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触发 器 


( 镭 视频 讲解 : 21 分 钟 ) 


触发 器 是 由 事件 来 触发 基 个 操作 。 这 些 事件 包括 INSERT 话 句 、 
UPDATE 语句 和 DELETE 语句 。 当 数据 库 系统 执行 这 些 事件 时 ， 就 会 
激活 触发 器 执行 相应 的 操作 。 本 章 将 介绍 触发 器 的 含义 和 作用 ， 可 以 
了 解 创 建 触发 器 、 坦 看 触发 器 和 删除 触发 器 的 方法 。 同 时 ， 还 可 以 了 
解 各 种 事件 的 触发 器 的 执行 情况 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

MH 了 解 MySQL 触发 器 的 概念 

由 了 解 在 MySQL 中 创建 单个 执行 语句 的 触发 路 

Wm 掌握 MySQL 中 创建 多 个 语句 的 触发 路 

MW 掌握 在 MySQL 数据 库 中 查看 触发 器 

MW 掌握 删除 触发 路 的 方法 

Mi 掌握 如 何 应 用 触发 路 
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17.1 MySQL 触发 器 


区 4 视频 讲解 : 光盘 \TM\Video\ 第 17 章 \MySQL 触发 器 .exe 

触发 器 是 由 MySQL 的 基本 命令 事件 来 触发 某 种 特定 操作 ， 这 些 基本 的 命令 由 INSERT、UPDATE、 
DELETE 等 事件 来 触发 某 些 特定 操作 。 满 足 触发 器 的 触发 条 件 时 ， 数 据 库 系 统 就 会 自动 执行 触发 器 中 定义 
的 程序 语句 。 这 样 可 以 令 某 些 操作 之 间 的 一 致 性 得 到 协调 。 


17.1.1 创建 MySQL 触发 器 


在 MySQL 中 ， 创 建 只 有 一 个 执行 语句 的 触发 器 的 基本 形式 如 下 : 
CREATE TRIGGER 触发 器 名 BEFORE | AFTER 触发 事件 
ON 表 名 FOR EACH ROW 执行 语句 


具体 的 参数 说 明 如 下 : 
触发 器 名 指定 要 创建 的 触发 器 名 字 。 
BEFORE 和 AFTER 指定 触发 器 执行 的 时 间 。BEFORE 指 在 触发 时 间 之 前 执行 触发 语句 ，AFTER 
表示 在 触发 时 间 之 后 执行 触发 语句 。 
触发 事件 指定 数据 库 操作 触发 条 件 ， 其 中 包括 INSERT\UPDATE 和 DELETE。 
表 名 指定 触发 时 间 操 作 表 的 名 称 。 
FOR EACH ROW 表示 任何 一 条 记录 上 的 操作 满足 触发 事件 都 会 触发 该 触发 器 。 
执行 语句 指定 触发 器 被 触发 后 执行 的 程序 。 
例 17.1 下 面 创建 一 个 由 插入 命令 “INSERT” 触 发 的 触发 器 auto_save_time。《〔 实 例 位 置 ， 光 盘 \TM\ 
Instance\17\17.1) 
具体 步骤 如 下 : 
(1) 创建 一 个 名 称 为 timelog 的 表格 ， 该 表 的 结构 非常 简单 。 相 关 代 码 如 下 : 
create table timelog( 
id int(11) primary key auto_increment not null, 
savetime varchar(50) not null 
): 


四 加 


加 回回 加 


(2) 创建 名 称 为 auto_save_time 的 触发 器 。 其 代码 如 下 : 
delimiter // 

create trigger auto_save_time before insert 

on studentinfo for each row 

insert into timelog(savetime) values(now!()); 

I 


以 上 代码 的 运行 结果 如 图 17.1 所 示 。 

auto_save_time 触发 器 创建 成 功 ， 其 具体 的 功能 是 当 用 户 向 studentinfo 表 中 执行 INSERT 操作 时 ， 数 据 
库 系统 会 自动 在 插入 语句 执行 之 前 向 timelog 表 中 插入 当前 时 间 。 下 面 通过 向 studentinfo 表 中 插入 一 条 信息 
来 查看 触发 器 的 作用 。 其 代码 如 下 : 


insert into studentinfo(name) values (Chris'); 
然后 执行 SELECT 语句 查看 timelog 表 中 是 否 执行 NSERT 操作 ， 其 结果 如 图 17.2 所 示 。 
以 上 结果 显示 ， 在 向 studentinfo 表 中 插入 数据 时 ，savetime 表 中 也 会 被 插入 一 条 当前 系统 时 间 的 数据 。 


局 


1 


图 17.1 创建 auto_save time 触发 器 图 17.2 查看 timelog 表 中 是 否 执行 插入 操作 
17.1.2 创建 具有 多 个 执行 语句 的 触发 器 


上 面 17.1.1 节 中 ， 已经 介绍 了 如 何 创建 一 个 最 基本 的 触发 器 ， 但 是 在 实际 应 用 中 ， 往 往 触发 器 中 包含 
多 个 执行 语句 。 其 中 创建 具有 多 个 执行 语句 的 触发 器 语法 结构 如 下 : 

CREATE TRIGGER 触发 器 名 称 BEFORE | AFTER 触发 事件 

ON 表 名 FOR EACH ROW 

BEGIN 

执行 语句 列表 

END 

其 中 ， 创 建 具有 多 个 执行 语句 触发 器 的 语法 结构 与 创建 触发 器 的 一 般 语法 结构 大 体 相 同 ， 其 参数 说 明 
请 参考 17.1.1 节 中 的 参数 说 明 ， 这 里 不 再 歼 述 了 。 在 该 结构 中 ， 将 要 执行 的 多 条 语句 放 入 BEGIN 与 END 
之 间 。 多 条 语句 需要 执行 的 内 容 ， 需 要 用 分 隔 符 “;” 隔 开 。 


LE 一 般 放 在 BEGIN 与 END 之 间 的 多 条 执行 语句 必 须 用 结束 分 隔 符 “:” 分 开 。 在 创建 触发 器 过 
程 中 需要 更 改 分 隔 符 ， 故 这 里 应 用 上 一 章 提 到 的 DELIMITERT 语句 ， 将 结束 符号 变 为 “//”。 当 触发 器 
创建 完成 后 ， 读 者 同样 可 以 应 用 该 语句 将 结束 符 摘 回 “;”。 


下 面 创建 一 个 由 DELETE 触发 多 个 执行 语句 的 触发 器 delete_time info。 模 拟 一 个 删除 日 志 数据 表 和 一 
个 删除 时 间 表 。 当 用 户 删 除数 据 库 中 的 某 条 记录 后 ， 数 据 库 系 统 会 自动 向 日 志 表 中 写 入 日 志 

例 17.2 在 上 例 中 创建 的 timelog 数据 表 基 础 上 , 另外 创建 一 个 名 称 为 timeinfo 的 数据 
光盘 \TM\Instance\17\17.2) 

创建 代码 如 下 : 

create table timeinfo( 

id int(11), primary key auto_increment, 

info varchar(50) not null 


。( 实 例 位 置 : 


)/ 
然后 创建 一 个 由 DELETE 触发 多 个 执行 语句 的 触发 器 delete_time info。 其 代码 如 下 : 
delimiter // 


create trigger delete_time_info after delete 

on studentinfo for each row 

begin 

insert into timelog(savetime) values (now()); 

insert into timeinfo(info) values (deleteact'); 

end 

I 

运行 以 上 代码 的 结果 如 图 17.3 所 示 。 

例 17.3 ”和 触发 器 创建 成 功 ， 当 执行 删除 操作 后 ， 
timelog 与 timeinfo 表 中 将 会 插入 两 条 相关 记录 。 执 行 
删除 操作 的 代码 如 下 : (实例 位 置 : 光盘 \TMNInstance\ 
17\17.3) 图 17.3 创建 具有 多 个 语句 的 触发 器 delete time info 
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DELETE FROM studentinfo where sid=7:; 
删除 成 功 后 ,应 用 SELECT 语句 分 别 查看 timelog 数据 表 与 timeinfo 数据 表 。 其 运行 结果 如 图 17.4 和 
图 17.5 所 示 。 


图 17.4 查看 timelog 数据 表 信 息 图 17.5 查看 timeinfo 数据 表 信 息 


从 图 17.4 和 图 17.5 中 可 以 看 出 ,触发 器 创建 成 功 后 , 当 用 户 对 students 表 执行 DELETE 操作 时 , students 
数据 库 中 的 timelog 数据 表 和 timeinfo 数据 表 中 分 别 被 插入 操作 时 间 和 操作 信息 。 


[a 技巧 在 MysQL 中 ， 一 个 表 在 相同 的 时 间 和 相同 的 能 发 时 间 只 能 创建 一 个 能 发 器 ， 如 触发 时 间 
INSERT 和 触发 时 间 AFTER 的 触发 器 只 能 有 一 个 。 但 是 可 以 定义 BEFORE 的 触发 器 。 


17.2 查看 触发 器 


区 视频 讲解 : 光盘 \TM\Video\ 第 17 章 \ 查 看 触发 器 .exe 
查看 触发 器 是 指 查看 数据 库 中 已 存在 的 触发 器 的 定义 、 状 态 和 语法 等 信息 。 查 看 触发 器 应 用 SHOW 
TRIGGERS 语句 。 


17.2.1 SHOW TRIGGERS 


在 MySQL 中 ， 可 以 执行 SHOW TRIGGERS 语句 查看 触发 器 的 基本 信息 。 其 基本 形式 如 下 : 

SHOW TRIGGERS; 

进入 MySQL 数据库， 选择 students 数据 库 并 查看 该 
数据 库 中 存在 的 触发 器 ， 其 运行 结果 如 图 17.6 所 示 。 

在 命令 提示 符 中 输入 SHOW TRIGGERS 语句 即 可 查 
看 选择 数据 库 中 的 所 有 触发 器 ， 但 是 ， 应 用 该 查看 语句 
存在 一 定 次 端 ， 即 只 能 查询 所 有 触发 器 的 内 容 ， 并 不 能 
着 定 查看 某 个 触发 器 的 信息 。 这 样 一 来 ， 就 会 在 用 户 查 
找 指定 触发 器 信息 时 带 来 极 大 不 便 。 故 推荐 读者 只 在 触 
发 器 数量 较 少 的 情况 下 应 用 SHOW TRIGGERS 语句 查询 
触发 器 基本 信息 。 


17.2.2 ”查看 triggers 表 中 触发 器 信息 


图 17.6 查看 触发 器 


在 MySQL 中 , 所 有 触发 器 的 定义 都 存在 该 数据 库 的 triggers 表 中 。 读者 可 以 通过 查询 triggers 表 来 查看 
数据 库 中 所 有 触发 器 的 详细 信息 。 查 询 语 句 如 下 : 

SELECT * FROM information_schema.triggers; 

其 中 , information_ schema 是 MySQL 中 默认 存在 的 库 , 而 information_ schema 是 数据 库 中 用 于 记录 触发 
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器 信息 的 数据 表 。 通 过 SELECT 语句 查看 触发 器 信息 。 其 运行 结果 与 图 17.6 相同 。 但 是 如 果 用 户 想 要 查看 
某 个 指定 触发 器 的 内 容 。 可 以 通过 where 子 句 应 用 TRIGGER 字段 做 为 查询 条 件 。 其 代码 如 下 所 示 : 
SELECT * FROM information_schema triggers WHERE TRIGGER_NAME= ' 触 发 器 名 称 '; 
其 中 ， 触 发 器 名 称 这 一 参数 为 用 户 指定 要 查看 的 触发 器 名 称 ， 和 其 他 SELECT 查询 语句 相同 ， 该 名 称 
内 容 需 要 用 一 对 “"” ( 单 引号 引用 指定 的 文字 内 容 。 


[a 技巧 如 果 数 据 库 中 存在 数量 较 多 的 触发 器 , 建议 读者 使 用 第 二 种 查看 触发 器 的 方式 . 这样 会 在 查找 
指定 触发 器 过 程 中 避免 很 多 不 必要 的 麻烦 。 


17.3 ”应 用 触发 器 


菇 4 视频 讲解 : 光盘 \TM\Video\ 第 17 章 \ 应 用 触发 器 .exe 
在 MySQL 中 ， 触 发 器 按 以 下 顺序 执行 : BEFORE 触发 器 、 表 操作 、AFTER 触发 器 操作 ， 其 中 表 操 作 
包括 常用 的 数据 库 操作 命令 ， 如 INSERT、UPDATE、DELETE。 
例 17.4 触发 器 与 表 操作 存在 执行 顺序 ， 下 面 通 过 创建 一 个 实例 向 读者 展示 3 者 的 执行 顺序 关系 。( 实 
例 位 置 ， 光盘 \TM\Instance\17\17.4) 
(1) 创建 名 称 为 before_in 的 BEFORE INSERT 触发 器 。 其 代码 如 下 : 
create trigger before_in before insert on 
studentinfo for each row 
insert into timeinfo (info) values (‘before'); 
(2) 创建 名 称 为 after in 的 AFTER INSERT 触发 器 。 其 代码 如 下 : 
create trigger after_in after insert on 
studentinfo for each row 
insert into timeinfo (info) values (‘after’); 
运行 步骤 (1) 、 (2) 的 结果 如 图 17.7 所 示 。 
(3) 创建 完毕 触发 器 ， 向 数据 表 studentinfo 中 插入 一 条 记录 。 代 码 如 下 : 
insert into studentinfo(name) values (Nowitzki); 
(4) 执行 成 功 后 ， 通 过 SELECT 语句 查看 tmeinfo 数据 表 的 插入 情况 。 代 码 如 下 : 
select * from timeinfo; 
运行 以 上 代码 ， 其 运行 结果 如 图 17.8 所 示 。 


图 17.7 创建 触发 器 运行 结果 图 17.8 ”查看 timeinfo 表 中 触发 器 的 执行 顺序 
查询 结果 显示 before 和 after 触发 器 被 激活 。before 触发 器 首先 被 激活 ， 然 后 after 触发 器 再 被 激活 。 


像 技巧 触发 器 中 不 能 包含 START TRANSCATION、COMMIT 或 ROLLBACK 等 关键 字 , 也 不 能 包含 
CALL 语句 。 触 发 器 执行 非常 严密 ， 每 一 环 都 息息相关 ， 任 何 错 误 都 可 能 导致 程序 无 法 向 下 执行 。 已 经 
更 新 过 的 数据 表 是 不 能 回 滚 的 。 故 在 设计 过 程 中 一 定 要 注意 触发 器 的 逻辑 严密 性 。 
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17.4 删除 触发 器 


铬 4 视频 讲解 : 光盘 \TM\Video\ 第 17 章 \ 删 除 触发 器 .exe 

在 MySQL 中 ， 既 然 可 以 创建 触发 器 ， 同 样 也 可 以 通过 命令 删除 触发 器 。 删 除 触发 器 指 删除 原来 已 经 在 
某 个 数据 库 中 创建 的 触发 器 ,与 MySQL 中 删除 数据 库 的 命令 相似 。 删 除 触发 器 应 用 DROP 关键 字 。 其 语法 
格式 如 下 : 

DROP TRIGGER 触发 器 名 称 

触发 器 名 称 参数 为 用 户 指定 要 删除 的 触发 器 名 称 ， 如 果 指 定 某 个 特定 触发 器 名 称 ，MySQL 在 执行 过 程 
中 将 会 在 当前 库 中 查找 触发 器 。 


[a 技巧 在 应 用 完 触 发 器 后 ， 切 记 一 定 要 将 触发 器 删除 ， 否 则 在 执行 菜 些 数据 库 操 作 时 ， 会 造成 数据 的 变化 。 


例 17.5 下 面 将 名 称 为 delete_time_info 的 触发 器 删除 。〔 实 例 位 置 ， 光盘 \TM\Instance\17\17.5) 
其 执行 代码 如 下 : 

DROP TRIGGER delete_time_info; 

执行 上 述 代码 ， 其 运行 结果 如 图 17.9 所 示 。 

通过 查看 触发 器 命令 来 查看 数据 库 students 中 的 触发 器 信息 。 其 代码 如 下 : 

SHOW TRIGGERS 

查看 触发 器 信息 ， 可 以 从 图 17.10 看 出 ， 名 称 为 delete_time_info 的 触发 器 已 经 被 删除 了 。 


图 17.9 删除 触发 器 图 17.10 查看 students 数据 库 中 的 触发 器 信息 
息 5 汗 
$9 注意 图 17 10 的 过 回 结果 显示 ， 该 数据 库 中 存在 两 个 触发 器 信息 ， 这 两 个 能 发 器 是 在 17.1.2 节 中 被 


创建 的 ， 如 果 用 户 在 db databasel17 数据 库 中 未 创建 该 触发 器 ， 则 返回 结果 会 是 一 个 “Empty set”。 


17.5 实 战 


17.5.1 创建 一 个 由 INSERT 触发 的 触发 器 
例 17.6 下 面向 department 表 中 插入 一 条 记录 , 然后 查看 tb_students 表 中 是 否 执 行 insert 操作 。( 实 例 
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位 置 : 光盘 \TM\Instance\17\17.6) 


代码 如 下 : 

CREATE TRIGGER trig1 BEFORE INSERT 

ON department FOR EACH ROW 

INSERT INTO tb_students(times) VALUES(NOW()); 
INSERT INO department(name) values(liliy ); 
Select * from tb_students; 


执行 上 述 代码 ， 其 运行 结果 如 图 17.11 所 示 。 
17.5.2 ”获取 数据 库 中 的 触发 器 


例 17.7 在 图 17.12 中 的 文本 框 中 输入 要 查看 数据 触发 器 , 然后 单 击 “ 查 看 ”按钮 即 可 将 该 数据 库 
中 所 有 的 触发 器 的 详细 信息 显示 出 来 。 效 果 如 图 17.12 所 (实例 位 置 ， 光盘 \TMNInstance\17\17.7) 


收取 数据 库 中 的 甬 发 内 
明日 科技 


71 ES 
EE 
HL dputreeat | INSERT 
图 17.11 创建 一 个 由 INSERT 触发 的 触发 器 17.12 ”获取 数据 库 中 的 触发 器 
实现 过 程 如 下 。 
(1) 选择 要 查看 数据 库 的 触发 器 。 代 码 如 下 : 
<?php 


$dbname=$_POST[name]; 
$conn=mysql_connect("localhost","root","111"); 
mysql_select_db($dbname,$conn); 
mysql_query("set names gb2312"); 
(2) 执行 show trigger 语句 并 显示 所 查找 到 的 触发 器 的 详细 信息 。 代 码 如 下 : 
<?php 
$sql=@mysql_query("show triggers", $conn); 
$info=@mysql_fetch_array($sql); 
if($info==false}{ 
?> 
<tr> 
<td height="25" colspan="5" bgcolor="#FFFFFF"><div align="center"> 该 数据 库 没有 设置 触发 器 !</div></td> 
</tr> 
<?php 
}yelse{ 
dof 
?> 
<tr> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo $info[ Trigger];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo $info[Table];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo S$info[Event];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo $info[Timing];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo $info['Statement];?></div></td> 
</tr> 
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<?php 
}while($info=mysql_fetch_array($sql)); 
} 


?> 


17.5.3 使 用 DROP TIRGGER 删除 触发 器 


例 17.8 删除 触发 器 指 删 除 原来 已 经 在 某 个 数据 库 中 创建 的 触发 器 , 与 MySQL 中 删除 数据 库 的 命令 相 
似 。 删 除 触发 器 应 用 DROP 关键 字 。 (实例 位 置 : 光盘 \TM\Instance\17\17.8) 

代码 如 下 : 

DROP TRIGGER trig1 

SHOW TRIGGERS:; 

执行 上 述 代 码 ， 删 除 前 后 的 效果 如 图 17.13 和 图 17.14 所 示 。 


图 17.13 ”删除 触发 器 之 前 图 17.14 ”删除 触发 器 之 后 


17.6。 水 结 


本 章 对 MySQL 数据 库 的 触发 器 的 定义 和 作用 、 创 建 触发 器 、 查 看 触发 器 、 使 用 触发 器 和 删除 触发 器 等 
内 容 进 行 了 详细 讲解 ， 创 建 触发 器 和 使 用 触发 器 是 本 章 的 重点 内 容 。 读 者 在 创建 触发 器 后 ， 一 定 要 查看 触 
发 器 的 结构 。 使 用 触发 器 时 , 触发 器 执行 的 顺序 为 BEFORE 触发 器 、 作 (INSERT、UPDATE 和 DELETE) 
和 AFTER 触发 器 。 读 者 需要 将 本 章 的 知识 结合 实际 需要 来 设计 触发 器 。 


17.7 学 习 成 果 检验 


1. 如 何 查看 触发 器 信息 ? 〔 实 例 位 置 ， 光盘 \TMN\Instance\17\17.9) 

2. 触发 器 的 触发 顺序 是 什么 ? 〔 实 例 位 置 : 光盘 \TMNInstance\17\17.10) 

3. 触发 器 执行 语句 的 限制 条 件 是 什么 ? 〔 实 例 位 置 ， 光盘 \TM\Instance\17\17.11) 

4. MySQL 中 创建 多 条 执行 语句 的 触发 器 总 是 遇 到 分 号 就 结束 创建 ， 然 后 报错 ， 为 什么 ? 实例 位 置 : 
光盘 \TM\Instance\17\17.12) 
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物流 管理 系统 


(种 视频 讲解 ，83 分 钟 ) 


综合 实例 三) 


物流 信息 化 是 指 物流 企业 运用 现代 信息 技术 对 物流 过 程 中 产生 的 
全 部 或 部 分 信息 进行 和 采集、 分类、 传递 、 汇 总 、 坦 询 等 一 系列 处 理 活 
动 ， 以 实现 对 货物 流动 过 程 的 控制 ， 从 而 降低 成 本 、 提 高 效益 的 管理 
活动 。 

物流 企业 信息 化 的 目的 就 是 要 满足 企业 自身 管理 的 需要 和 不 同类 
型 企业 在 物流 业务 外 包 过 程 中 对 信息 交换 的 要 求 ， 也 就 是 通过 建设 物 
流 信 息 系统 ， 提 高 信息 流转 效率 ， 降 低 物 流 运 作成 本 。 

目前 我 国正 处 于 全 面 推进 信息 化 的 进程 中 ， 所 以 物流 领域 的 信息 
化 已 成 为 一 个 必然 。 物 流 信息 化 将 成 为 现代 物流 的 灵 现 ， 将 是 现代 物 
流 发 展 的 必 经 之 路 。 

通过 阅读 本 章 内容 ， 你 可 以 : 

NM 了 解 物流 配送 系统 开发 的 基本 过 程 

Wm 熟悉 如 何 创 建 MySQL 数据 库 的 存储 过 程 

让 熟悉 如 何 通过 MySQL 数据 库存 储 过 程 实现 用 户 登 录 

让 熟悉 如 何 实现 数据 库 中 数据 的 模糊 查询 技术 

NM 熟悉 如 何 创 建 浮动 框架 

WI 熟悉 如 何 实现 订单 的 打印 技术 


PHP+MySQL 开发 实战 


18.1 物流 管理 系统 概述 


随 着 我 国信 息 化 进程 的 全 面 推进 ， 各 领域 的 信息 化 进程 都 在 飞速 发 展 ， 同 样 也 推动 着 物流 领域 的 信息 
化 进程 飞快 地 向 前 发 展 。 由 于 信息 化 进程 的 全 面 推进 ， 对 现代 物流 提出 更 高 的 要 求 : 信息 化 、 自 动 化 、 网 
络 化 、 智 能 化 和 人 性 化 等 。 物 流行 业 的 竞争 日 益 激烈 ， 客 户 需 求 的 标准 也 越 来 越 高 ， 物 流 企业 要 想 在 市 场 
中 占有 一 席 之 地 ， 必 须要 建立 一 个 高 效 、 快 捷 、 方 便 的 物流 信息 系统 ， 为 客户 提供 一 流 的 服务 ， 并 且 能 够 
及 时 准确 地 掌握 客户 的 需求 ， 对 客户 的 需求 做 出 快速 反映 ， 在 最 短 的 时 间 内 以 最 大 限度 挖 据 和 优化 物流 资 
源 来 满足 客户 的 需求 ， 从 而 建立 高 效 的 数字 化 物流 经 济 。 


18.2 ”系统 分 析 流 程 


18.2.1 需求 分 析 


随 着 现代 物流 信息 化 进程 的 加 快 ， 传 统 的 物流 管理 方式 已 经 不 再 适应 当前 物流 发 展 的 要 求 ， 取 而 代 之 
的 将 是 以 计算 机 为 基础 的 网 络 化 物流 管理 方式 。 某 物流 配送 公司 为 适应 物流 信息 化 进程 的 发 展 ， 急 需 开发 
-个 物流 配送 系统 ， 通 过 网 络 来 实现 对 物流 操作 流程 进行 管理 ， 不 但 为 企业 的 运营 过 程 节省 大 量 的 人 力 、 
物力 、 财 力 和 时 间 ， 提 高 物流 系统 运行 的 效率 ， 而 且 为 企业 在 客户 中 树立 一 个 全 新 的 形象 ， 为 企业 的 发 展 
商定 一 个 良好 的 基础 。 现 根据 对 该 物流 配送 公司 的 实际 调查 ， 以 及 公司 的 具体 要 求 ， 制 定 出 物流 管理 系统 
的 规划 方案 。 具 体内 容 如 下 
网 站 页 面 设计 要 求 美观 大 方 ， 能 够 展示 企业 形象 。 
为 企业 在 客户 中 树立 一 个 全 新 的 形象 。 
网 站 的 操作 流程 简单 、 方 便 ， 能 够 提高 工作 效率 。 
提供 物流 配送 的 全 程 跟踪 。 
提供 配送 信息 的 及 时 查询 。 
实现 对 配送 车 辆 的 管理 。 
实现 对 客户 信息 的 管理 。 
实现 发 货 单 打印 的 功能 。 


18.2.2 可行 性 分 析 


因 因 办 办 办 办 罗 


物流 管理 系统 的 开发 不 但 能 使 物流 企业 走 上 科学 化 、 网 络 化 管理 的 道路 ， 而 且 能 够 为 企业 带 来 巨大 的 
经 济 效益 和 技术 上 飞速 的 发 展 。 

1. 经 济 性 

科学 的 管理 方法 ， 便 捷 的 操作 环境 ， 系 统 的 经 营 模式 ， 将 为 企业 带 来 更 多 的 客户 资源 ， 树 立 企业 的 品 
牌 形象 ， 提 高 企业 的 经 济 效益 。 

2. 技术 性 

网 络 化 的 物流 管理 方式 ， 在 操作 过 程 中 能 够 快捷 地 查找 出 车 源 信息 、 客 户 订单 以 及 客户 信息 ; 能够 对 


后 
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货物 进行 全 程 跟踪 ， 了 解 货物 的 托运 情况 ， 从 而 使 企业 能 够 根据 实际 情况 ， 做 好 运营 过 程 中 的 各 项 准备 工 
作 ， 作 出 及 时 准确 的 调整 ， 能够 保证 托运 人 以 及 收 货 人 对 货物 进行 及 时 地 处 理 。 


18.3 系统 设计 流程 


18.3.1 系统 目标 


结合 目前 网 络 上 物流 配送 系统 的 设计 方案 ， 对 客户 做 的 调查 结果 以 及 企业 的 实际 需求 。 本 项 目 在 设计 
时 应 该 满足 以 下 目标 : 
界面 设计 美观 大 方 、 操 作 简单 。 
功能 完善 、 结 构 清晰 。 
能 够 快速 查询 车 源 信息 。 
能 够 准确 填写 订单 。 
能 够 实现 订单 查询 、 打 印 。 
能 够 实现 对 回 单 处 理 。 
能 够 对 车 源 信息 进行 添加 、 修 改 和 删除 。 
能 够 对 客户 信息 进行 管理 。 
能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 
良好 的 数据 库 系统 支持 。 
系统 运行 稳定 ， 具 备 良好 的 防范 措施 。 


18.3.2 ”系统 功能 结构 
结合 需求 分 析 和 系统 目标 中 的 内 容 ， 物 流 管理 系统 的 功能 结构 已 经 设计 完成 。 为 了 使 读者 能 够 更 清楚 


地 了 解 网 站 的 结构 ， 下 面 给 出 物流 管理 系统 的 功能 模块 结构 图 和 工作 流程 图 。 
物流 管理 系统 的 功能 模块 结构 图 ， 如 图 18.1 所 示 。 


加 图 国峰 回回 加 图 罗 加 网 


删除 车 源 信息 


图 18.1 物流 管理 系统 功能 结构 图 
物流 管理 系统 的 工作 流程 图 如 图 18.2 所 示 。 
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I 客户 提交 订单 
P| 
(OO 


客户 


加 车 辆 查询 
(A 
回 


< 
re | 填写 发 代 音 


货物 发 出 


图 18.2 物流 管理 系统 的 工作 流程 图 


18.3.3 ”系统 预览 


物流 管理 系统 由 多 个 程序 页 面 组 成 ， 下 面 给 出 几 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 物 流 管 
理 系统 主页 如 图 18.3 所 示 ， 该 页 面 用 于 展示 本 系统 的 车 源 信息 查询 模块 。 登 录 页 面 如 图 18.4 所 示 。 
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图 18.3 主页 图 18.4 登录 页 面 


回执 发 货 单 确认 模块 的 页 面 效果 如 图 18.5 所 示 ， 该 页 面 主要 用 于 实现 对 回执 发 货 单 的 处 理 。 发 货 单 查 
询 模块 的 页 面 效果 如 图 18.6 所 示 ， 该 页 面 主要 用 于 实现 对 指定 发 货 单 的 查询 ， 并 且 输 出 发 货 单 的 详细 内 容 。 
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图 18.5 回执 发 货 单 图 18.6 发 货 单 查询 
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车 源 信息 管理 模块 的 页 面 效 果 如 图 18.7 所 示 ， 该 页 面 实现 对 车 源 信息 的 添加 、 修 改 和 删除 操作 。 修 改 
管理 员 密码 模块 的 页 面 效果 如 图 18.8 所 示 ， 该 页 面 实现 对 管理 员 密 码 的 修改 。 


“Ma a 
i es 


图 18.7 车 源 信息 管理 图 18.8 修改 管理 员 密码 


18.4 数据 库 设 计 


敬 i 视频 讲解 : 光盘 \TMNVideo\ 第 18 章 \ 数 据 库 设计 .exe 

物流 管理 系统 必须 拥有 数据 库 的 支持 ， 所 有 物流 配送 的 数据 都 应 该 存储 到 数据 库 中 ， 便 于 管理 员 查 找 
车 辆 、 订 单 和 客户 的 信息 。 如 果 没 有 数据 库 的 支持 ， 那 么 物流 管理 系统 将 没有 任何 意义 。 本 节 将 对 物流 管 
理 系统 的 数据 库 设 计 进 行 详细 介绍 。 


18.4.1 数据 库 分 析 


物流 管理 系统 是 一 个 中 小 型 的 企业 管理 系统 ， 考 虑 到 开发 的 成 本 、 搭 配 的 合理 性 以 及 操作 的 灵活 性 等 ， 
使 用 MySQL 数据 库 是 最 佳 的 选择 。MySQL 数据 库 是 完全 免费 的 ， 使 用 它 不 需要 任何 费用 ， 可 以 直接 从 网 
上 免费 下 载 ; MySQL 数据 库 的 操作 也 非常 方便 ， 不 但 可 以 在 命令 模式 下 操作 ， 而 且 配 备 目前 比较 流行 的 图 
形 化 管理 工具 phpMyAdmin， 能 够 轻松 地 实现 对 MySQL 数据 库 的 管理 和 操作 。 


18.4.2 ”数据 库 概 念 设计 


根据 上 述 各 节 对 物流 管理 系统 做 的 需求 分 析 和 系统 设计 ， 整 理 出 物流 管理 系统 的 实体 关系 E-R 图 。 其 
中 包括 管理 员 信息 实体 、 车 源 信 息 实 体 、 车 辆 日 志 信 息 实体 、 客 户 信息 实体 和 发 货 单 信息 实体 。 

1. 管理 员 信息 实体 

管理 员 信息 实体 用 于 存储 管理 员 的 登录 名 称 和 密码 信息 ， 包 括 用 户 名 和 密码 两 项 内 容 。 管 理 员 信息 实 
体 的 E-R 图 如 图 18.9 所 示 。 


用 户 名 管理 员 


18.9 管理 员 信息 实体 的 E-R 图 
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2. 车 源 信息 实体 
车 源 信息 实体 用 于 存储 企业 拥有 的 车 辆 信息 ， 包 括 车 主 姓名 、 车 主 身份 证 号 码 、 车 牌号 码 、 车 主 联 系 
电话 、 车 主 的 家 庭 地 址 、 车 辆 行驶 的 路 线 和 车 辆 描述 。 车 源 信息 实体 的 E-R 图 如 图 18.10 所 示 。 


图 18.10 车 源 信息 实体 的 E-R 图 
3. 车 辆 日 志 信 息 实体 


车 辆 日 志 信息 实体 记录 车 辆 当前 是 否 被 占用 ， 以 及 使 用 的 时 间 、 执 行 的 任务 ， 包 括 车 牌号 码 、 车 辆 日 
志 信息 《详细 的 车 辆 使 用 描述 ) 、 日 志 时 间 和 发 货 单 ID 〈 即 执行 的 任务 ) 。 车 辆 日 志 信息 实体 的 E-R 图 如 


图 18.11 所 示 。 
图 18.11 车 辆 日 志 信息 实体 的 E-R 图 
4. 客户 信息 实体 


客户 信息 实体 用 于 存储 客户 的 信息 ， 包 括 客户 的 姓名 、 客 户 电话 和 客户 的 联系 地 址 。 客 户 信息 实体 的 
E-R 图 如 图 18.12 所 示 。 


客户 联系 地 址 


图 18.12 客户 信息 实体 E-R 图 


5. 发 货 单 信息 实体 
发 货 单 信息 实体 用 于 存储 客户 填写 的 发 货 单 信息 ， 包 括 车 牌号 码 、 车 主 电话 、 货 物 描述 、 发 货 人 、 发 
货 时 间 、 发 货 人 电话 、 发 货 地 址 、 收 货 人 、 收 货 人 电话 、 收 货 地 址 和 付款 方式 。 发 货 单 信息 实体 的 E-R 图 


如 图 18.13 所 示 。 
发 货 人 
rT > 
CC 车主 电话 发 货 单 收 货 地 址 ”》 


18.13 ”发 货 单 信息 实体 的 E-R 
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18.4.3 ”创建 数据 库 及 数据 表 


在 物流 管理 系统 中 应 用 的 是 db_database18 数据 库 ， 其 中 涉及 5 个 数据 表 ， 如 图 18.14 所 示 。 
加 服务 器 :localhost 、 居 数据 库 : db_database18 
Ei 

到 和 记录 教 轨 型 要 于 大 小。 多 未 


万 moem 国 四 加 苇 面 X 9 MsAy gun312chinese_rl 。 28KB 四 

记 we 国 盏 轩 革 轩 xX 4 NYSAY go2312_chnese_s| 25K9 188 享 节 

已 werkg 辣 除 轩 革 硬 X 1 NEAY go2312 chinese_s1 26KB 四 宁 节 

CC tcustomer 百 加 节 午 XxX 16 MySAY go2312 chinese | 26KB 20 

上 tbshoppme 司 东 加 荚 重 尖 17 MISAW gb2312_chnese_rj 46KB 260 字 六 
图 18.14 ”物流 管理 系统 的 数据 库 


下 面 介绍 物流 管理 系统 中 使 用 的 数据 表 ， 数 据 表 的 设计 结构 如 图 18.15 一 图 18.19 所 示 。 
1. tb_admin (管理 员 信 息 表 ) 
管理 员 信息 表 主 要 用 于 存储 管理 员 的 登录 名 和 密码 。 该 数据 表 的 结构 如 图 18.15 所 示 。 


园 服务 器 : localhost ， 局 数据 库 : db_database18 加 表 : tb_admin 
辐 测 呈 加 炎 构 “于 SQL 万 接 过 于 贡 入 加 导 出 脱 mpor。 猴 针 作 喀 潮 32 
字 度 天 时 要 各 司 性 Nd 加 认 开外 


本 人 
二 世 ia EE somwment 国 少 X 几 加 中 
tm ent waem enzmmesaal 村 量 次 X 国 四 吧 辐 
admn pass vorchar(S0) wo2312 chnese ei 否 轩 帮 X 轩 加 史 网 


图 18.15 管理 员 信息 表 结构 
2. tb_car (车 源 信息 表 ) 
车 源 信息 表 主要 用 于 存储 配送 公司 拥有 的 车 辆 信息 ， 主 要 包括 车 主 姓名 、 车 主 身份 证 号 码 、 车 牌号 码 、 
车 主 联系 电话 、 车 主 的 家 庭 地 址 、 车 辆 行驶 的 路 线 和 车 辆 描述 。 该 数据 表 的 结构 如 图 18.16 所 示 。 


中 服务 器 : lecalhost 局 数据 库 : db_database19 ) 四 表 : 岂 _car 
阳 间 的 了 区 移 , 避 Sol 广 相 来 敌 锋 入 、 卫 Ss 虹 wport 伏 拓 人 古寺 宇 。 吕 内 


3 类 型 要 里 属性 Ml 时 认 吴 外 手 作 
[a IO) 于 auomrement 阳 贿 X 图 网 吕 厅 
门 wsemame 。 varsher(50) 号 23312_srineea_ol 要 则 六 Xi 国 风 罗网 

二 莘 了 XX 国 吕 于 国 
理 攻克 X 国 区 双双 
要 草 尖 XX 图 凤 加 区 
下 硬 忆 XX 加 加 区 
二 攻关 xx 团 凤 加 加 
否 La 


图 18.16 车 源 信息 表 结 构 
3. tb_car_log 〈 车 辆 日 志 信息 表 ) 


车 辆 日 志 信息 表 主 要 用 于 存储 车 辆 的 使 用 情况 信息 ， 主 要 包括 车 牌号 码 、 车 辆 使 用 日 志 、 日 志 创建 时 
间 和 发 货 单 ID。 该 数据 表 的 结构 如 图 18.17 所 示 。 


表 服务 器 :localhost ， 导 数据 库 : db_database 夫 > 目 表 ;tb_car_log 
MR 下部 鬼 nso 上 广 扫 了 括 和 中 叶 :了 mpor 允 可 作 大 者 衬 。 医 员 
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图 18.17 车 辆 日 志 信 息 表 结构 
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4. tb_customer (客户 信息 表 ) 
客户 信息 表 主 要 用 于 存储 客户 的 信息 , 包括 姓名 、 电 话 以 及 联系 地 址 。 该 数据 表 的 结构 如 图 18.18 所 示 。 


图 服务 器 :Iocalhost ， 局 基 拓 庄 : db_dztabase18 ， 目 表 :也 _eustemer 
于 本 区 .sa 二 关 表 床 丘 和 卫 Sb 卫 mpet 兴 基 作 而 居 M 


3 ba a Na 要 让 二 人 
DD eater mm 到 ao meeret 于 少 X 辐 网 蝇 了 了 
站 ememermer 。 rernatol ma2a2chmessaa 到 司 产 X 加 站台 同 
DD emtomer tet washafl gl2ohnoooe 到 EE 
eumtormer_ aross vachar(e0) ge2313.chinoses 天 村 六 X 司 加 风口 


图 18.18 客户 信息 表 结 构 


5. tb_shopping (发 货 单 信息 表 ) 

发 货 单 信息 表 主 要 用 于 存储 客户 填写 的 发 货 单 中 的 信息 ， 主 要 包括 车 牌号 码 、 车 主 电话 、 货 物 描述 、 
发 货 人 、 发 货 时 间 、 发 货 人 电话 、 发 货 地 址 、 收 货 人 、 收 货 人 电话 、 收 货 地 址 和 付款 方式 。 该 数据 表 的 结 
构 如 图 18.19 所 示 。 


团 服务 器 :lecalhost ， 司 粕 据 库 ; db_database18 ， 目 表 :由 _shepplng 
_ 轿 浊 里“ 苛 寻 才 加 SoL 万 持 开放 入 呵 叶 出 。 吕 meor 入 量 作 硬 击 空 于 了 
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单 信息 表 结 构 
18.5 网 站 首页 设计 
茹 4 视频 讲解 : 光盘 \TM\Video\ 第 18 章 \ 网 站 首页 设计 .exe 


18.5.1 网 站 首页 概述 


网 站 首页 是 整个 网 站 的 门面 ， 是 浏览 者 看 到 的 第 一 视觉 界面 ， 所 以 在 设计 网 站 的 首页 时 应 该 将 网 站 中 主 
要 的 内 容 尽量 展示 给 浏览 者 ， 让 浏览 者 能 够 更 快 地 了 解 网 站 的 内 容 。 物 流 管 理 系统 的 首页 主要 包括 如 下 功能 。 
回 ”车 源 信息 查询 : 主要 实现 车 辆 信息 查询 ， 及 时 为 客户 选择 货物 配送 的 方案 。 
发 货 单 : 主要 用 于 客户 发 货 单 信息 填写 。 
回 ”回执 发 货 单 确认 : 主要 用 于 对 货物 配送 回执 单 的 确认 。 
回 ”发 货 单 查询 : 主要 用 于 对 发 货 单 进行 查询 。 
回 客户 信息 管理 : 主要 用 于 对 客户 的 信息 进行 管理 。 
回 
回 
下 


车 源 信息 管理 : 主要 用 于 对 企业 拥有 的 车 辆 信息 进行 管理 。 
修改 密码 : 主要 用 于 对 管理 员 登 录 的 密码 进行 修改 。 
` 面 看 一 下 本 案例 中 提供 的 网 站 首页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\18\index.php， 如 图 18.20 所 示 。 
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18.5.2 ”网 站 首页 设计 技术 


物流 管理 系统 是 为 物流 配送 公司 设计 的 一 个 物流 管理 系统 ， 其 页 面 的 设计 突出 简洁 、 方 便 ， 功 能 的 实现 
以 便于 操作 和 维护 为 根本 。 网 站 首页 的 页 面 设计 如 图 18.20 所 示 ， 由 一 个 简单 的 框架 组 成 ， 其 结构 由 两 部 分 组 
成 : 一 部 分 用 于 输出 网 站 中 包括 的 模块 , 另 一 部 分 用 于 输出 对 应 模块 中 的 内 容 。 首页 框架 结构 如 图 18.21 所 示 。 
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图 18.20 ”物流 管理 系统 首页 图 18.21 首页 框架 结构 
这 里 使 用 的 框架 结构 不 是 frame 框架 ， 而 是 应 用 switch 语句 ， 加 上 表格 的 霸 套 实现 的 框架 效果 。 


18.5.3 ”网 站 首页 的 实现 过 程 


物流 管理 系统 的 首页 设计 非常 简单 ， 主 要 应 用 一 个 简单 的 框架 功能 ， 并 且 以 超 链接 作为 辅助 ， 最 终 实 
现在 不 同 功能 模块 之 间 的 跳 转 。 

首先 连接 数据 库 ， 判 断 登 录用 户 的 身份 ， 如 果 是 管理 员 则 可 以 进入 网 站 进行 操作 ， 和 否则 将 提示 “请 您 
正确 登录 ! ”， 并 且 跳 转 到 网 站 的 登录 页 面 。 其 关键 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\18\indexs.php) 


<?php session_start(); /初始 化 session 变量 

include("conn/conn.php"); /连接 数据 库 

if($_SESSION[‘admin_user]==trueX{ /判断 登录 用 户 的 身份 

和 

<!-> 省 略 了 部 分 代码 <!--> 

<?php 

}else{ /没有 以 管理 员 登 录 ， 则 跳 转 到 登录 页 
echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>"; 

, 

2 


然后 设计 首页 中 使 用 的 模块 超 链接 ， 并 且 为 每 个 超 链接 定义 变量 标识 ， 当 单 击 不 同 功能 模块 超 链接 时 
提交 不 同 的 变量 标识 ， 作 为 下 一 步 获取 模块 中 详细 内 容 的 依据 ; 其 中 还 使 用 ononMouseOver 和 onMouseOut 


事件 ， 控 制 鼠 标 移动 到 和 离开 单元 格 时 显示 不 同 的 颜色 。 部 分 代码 如 下 : 
< 上 - -一 一 一 一 一 一 应 用 onMouseOver 和 onMouseOut 事 件 控制 鼠标 移 过 和 离开 时 的 时 间 一 一 一 一 一 一 一 一 -> 
<tr onMouseOver="this.bgColor=#9FCB3A"onMouseOut="this.bgColor=#F5F5F5" > 
®@ 
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<td height="28" align="center">&nbsp;</td> 
< 上 一 一 一 一 一 一 设置 车 源 信息 查询 模块 的 超 链接 ， 变 量 标识 为 "mbs= 车 源 信息 查询 "一 一 一 一 一 一 一 > 
<td align="left"><a href="indexs.php?Imbs= 车 源 信息 查询 "> 车 源 信息 查询 </a></td> 

</tr> 

<tr onMouseOver="this.bgColor=#9FCB3A"onMouseOut="this.bgColor=#F5F5F5" > 
<td height="28" align="center">&nbsp;</td> 
< 一 一 一 一 一 一 一 设置 发 货 单 模块 的 超 链接 ， 变 量 标识 为 "Imbs= 发 货 单 "一 一 一 一 一 一 一 一 一 
<td align="left"><a href="indexs.php?Imbs= 发 货 单 "> 发 货 单 </a></td> 

</tr> 

最 后 应 用 switch 语句 ， 根 据 变量 标识 $lmbs 提交 的 值 进 行 判 断 ， 使 用 include 包含 语句 调用 不 同 功能 模 

块 的 脚本 文件 。 其 关键 代码 如 下 : 


<?php 
switch($Imbs){ // 获 取 变 量 $Imbs 的 值 
case "车 源 信息 查询 ": /判断 当 变 量 $imbs 的 值 是 “车 源 信息 查询 ”时 输出 下 面 的 内 容 
include("car_select.php"); // 调 用 指定 的 包含 文件 
break; /跳出 循环 
// 这 里 省 略 了 部 分 代码 
case ™: /判断 当 变 量 的 值 为 空 时 
include("car_select.php"); // 当 变量 的 值 为 空 时 则 调用 该 文件 
break; // 跳 出 循环 
?> " 


18.6 车 源 信 息 查 询 模 块 设计 
铬 i 视频 讲解 光盘 \TM\Video\ 第 18 章 \ 车 源 信息 查询 模块 设计 .exe 


18.6.1 车 源 信息 查询 模块 概述 


车 源 信息 查询 模块 的 功能 是 根据 客户 提供 的 货物 配送 的 地 点 ， 从 数据 库 中 查询 有 关 该 路 线 的 车 辆 ， 为 
客户 提供 一 个 合理 配送 的 路 线 ， 最 后 确定 发 货 订 单 。 操 作 流程 如 图 18.22 所 示 。 


18.22 车 源 信息 查询 的 操作 流程 


18.6.2 ”模糊 查询 技术 
车 源 信息 查询 模块 的 主要 功能 就 是 查询 车 辆 的 使 用 信息 ， 为 客户 提供 最 合适 的 货物 配送 路 线 。 其 中 应 


@ 
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用 的 关键 技术 自然 就 是 查询 方法 ， 为 了 给 客户 提供 最 合适 、 最 满意 、 最 快捷 的 配送 服务 ， 这 里 使 用 的 是 模 
糊 查询 技术 。 

通过 模糊 查询 技术 ， 只 要 客户 提出 配送 货物 的 出 发 地 点 和 到 达 地 点 ， 管 理 员 就 可 以 从 数据 库 中 提取 出 
所 有 与 该 路 线 相关 的 车 辆 信息 ， 包 括 车 辆 的 承载 能 力 、 车 辆 是 否 被 占用 和 车 辆 的 使 用 情况 ， 都 一 目 了 然 ， 
客户 可 以 根据 实际 的 情况 〈 时 间 、 货 物 数量 、 路 线 等 ) 选择 车 辆 ， 确 认 后 填写 发 货 单 。 

模糊 查询 技术 的 实现 是 在 car_selectphp 文件 中 完成 的 ， 首 先 要 与 数据 库 建立 连接 ， 然 后 创建 一 个 form 
表单 ， 设 置 两 个 文本 框 selectl 和 select2 用 于 提交 出 发 地 点 和 到 达 地 点 ， 将 表单 的 值 提交 到 当前 页 ， 最 后 编 
写 PHP 脚本 语句 ， 以 表单 中 获取 的 数据 为 条 件 ， 执 行 like 模糊 查询 ， 从 数据 库 的 指定 表 中 查询 符合 条 件 的 
信息 ， 并 且 将 查询 出 的 信息 显示 到 页 面 中 。 程 序 的 关键 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\18\car_select.php) 


<?php 
session_start(); include("conn/conn.php"); // 连 接 数 据 库 
?> 
< 一 一 一 一 一 一 一 一 一 一 一 创建 form 表单 ， 提 交 出 发 地 点 和 到 达 地 点 一 一 一 一 一 一 一 一 一 一 一 > 


<form id="form1" name="form1" method="post" action="indexs.php?lImbs= 车 源 信息 查询 " onSubmit="javascript: 
return check_select_car():"> 
<td height="30" colspan="4" align="right" bgcolor="#FFFFFF"> 查 询 
© <input name="select1" type="text" id="select1" size="10"> 至 
© <input name="select2" type="text" id="select2" size="10"></td> 
<td colspan="2" bgcolor="#FFFFFF"><input type="submit" name="Submit" value=" 提 交 " /></td> 
</form> 
< 上 一 一 一 一 一 一 获取 表单 提交 的 值 ， 根 据 该 条 件 执行 like 模糊 查询 ， 从 数据 库 中 读 取 数 据 一 一 一 一 -一 一 -> 
<?php 
日 “if$_POST[select1]==true || $_POST[select2]==true){ /判断 表单 提交 的 值 是 否 为 真 
/* wxaxaxhthxhxwtwtttt 凡 表单 提交 的 值 为 条 件 ， 从 数据 表 中 查询 数据 essxxwxztwrwtwt */ 
@ $query="select* from tb_car where car_road like '%$select1%' and car_road like '%$select2%"; 


$result=mysql_query($query); /执行 查询 语句 
if(mysql_num_rows($result)>0X{ /| 判断 是 否 存在 符合 条 件 的 数据 
while($myrow=mysql_fetch_array($result)X{ /应 用 while 循环 语句 ， 输 出 符合 条 件 的 数据 
?> 
<tr> 


<td height="26" align="center" bgcolor="#FFFFFF"><?php echo $myrow[fcar_number];?></td> 
<td bgcolor="#FFFFFF"><?php echo $myrow['car_road'];?></td> 

<td colspan="2" bgcolor="#FFFFFF"><?php echo $myrow['car_content"];?></td> 

<td bgcolor="#FFFFFF"> 


<?php 
A******xx*** 根 据 车 牌号 码 ， 从 车 辆 日 志 信 息 表 中 读 取 对 应 车 辆 的 使 用 信息 ********* */ 
$querys=mysql_query("select * from tb_car_log where car_number='$myrow['car_number’]"); 
$myrows=mysql_fetch_array($querys); 


echo $myrows['car_ log]; /输出 车 辆 的 使 用 日 志 信息 
?> 
</td> 
<td align="center" bgcolor="#FFFFFF"> 
<?php 
if($myrows['car_log]==null}{ // 判 断 车 辆 日 志 是 否 为 空 ， 如 果 为 空 ， 则 执行 下 面 的 内 容 
echo "<a href='indexs.php?Imbs= 发 货 单 &ljid=$myrowlid]'> 使 用 该 车 </a>"; 
}else{ // 如 果 车 辆 日 志 不 为 空 ， 则 执行 下 面 的 内 容 
echo "<a href='indexs.php?Imbs= 发 货 单 &ljid=$myrow[lid]'> 预 定 该 车 </a>"; 
} 
?> 
</td> 
</tr> 


<?php }}else{ echo "您 查找 的 路 线 不 存在 ""; }}?> 


图 
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a 
和 @ select1: 指定 出 发 地 点 。 


@ select2: 指定 到 达 地 点 。 
四 证 语 和 句 : 判断 从 表单 中 获取 的 变量 是 否 为 真 。 
@ like: 执行 模糊 查询 ， 从 数据 库 中 读 取 符合 条 件 的 数据 。 


18.6.3 ”车 源 信息 查询 模块 的 实现 过 程 


车 源 信息 查询 模块 的 功能 是 根据 客户 提供 的 信息 ， 从 数据 库 中 读 取 有 关 某 个 路 线 的 车 辆 使 用 情形 ， 及 
时 为 客户 确定 一 个 合理 的 物流 配送 方案 。 该 模块 的 运行 结果 如 图 18.23 所 示 。 


$ 入 基 扩 


徐 物 汽 管理 系统 


图 18.23 车 源 信息 查询 模块 的 运行 结果 


车 源 信 息 查询 功能 的 实现 通过 car_select.php 文件 来 完成 ， 首 先 与 数据 库 建立 连接 ， 然后 创建 一 个 form 
表单 ， 将 客户 的 地 址 信息 提交 到 当前 页 ， 作 为 查询 的 条 件 ， 最 后 编写 PHP 脚本 语句 ， 以 表单 中 获取 的 地 点 
数据 为 条 件 ， 执 行 like 模糊 查询 ， 从 数据 库 的 指定 表 中 查询 符合 条 件 的 信息 ， 并 且 将 查询 出 的 信息 显示 到 
页 面 中 。 其 关键 代码 可 以 参考 光盘 \TM\18\car_select.php 文件 。 


18.7 发 货 单 管理 模块 设计 
茹 4 视频 讲解 : 光盘 \TIM\Video\ 第 18 章 \ 发 货 单 管理 模块 设计 .exe 


18.7.1 发 货 单 管理 模块 概述 


发 货 单 管理 模块 主要 包括 发 货 单 的 填写 、 发 货 单 查询 、 发 货 单打 印 和 发 货 单 删除 。 发 货 单 管理 模块 的 
主要 功能 如 图 18.24 所 示 。 


344 
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管理 员 


图 18.24 发 货 单 管理 模块 的 功能 结构 图 


18.7.2 ”发 货 单 编号 生成 技术 


如 何 才能 发 挥 函数 的 最 大 作用 ? 设计 者 不 但 能 够 从 表面 理解 函数 的 功能 和 作用 ， 而 且 还 能 够 以 一 种 发 
散 的 思维 方式 去 考虑 问题 ， 不 要 被 其 本 身 的 功能 所 束缚 ， 要 将 函数 中 某 些 功能 有 机 地 结合 起 来 ， 从 而 发 挥 
其 更 大 的 作用 。 

在 发 货 单 管理 模块 中 应 用 到 一 个 发 货 单 的 编号 ， 该 编号 是 发 货 单 的 唯一 标识 ， 不 允许 存在 重复 。 在 编 
写生 成 这 个 编号 的 程序 时 ， 首 选 方案 是 应 用 随机 函数 获取 随机 的 数字 作为 编号 ， 这 是 一 个 最 直接 的 做 法 ， 
但 是 考虑 到 发 货 单 编号 不 但 具有 唯一 性 ， 而 且 还 要 使 其 具有 一 定 的 标志 性 ， 从 这 个 角度 考虑 使 用 随机 函数 
就 有 些 不 适合 ， 因 为 其 不 具备 一 定 的 标志 性 。 这 时 ， 如 果 应 用 date0 函 数 则 可 以 达到 上 述 所 说 的 要 求 ， 不 但 
编号 具有 唯一 性 ， 而 且 还 有 规律 可 循 。 

为 了 能 够 更 好 地 理解 这 种 发 散 思维 的 编程 思想 ， 首 先 介 绍 一 下 date0 函 数 。 语 法 格式 如 下 : 

string date ( string format , int timestamp) 

该 函数 返回 将 参数 timestamp 按照 指定 格式 字符 串 而 产生 的 字符 串 ， 其 中 的 参数 timestamp 是 可 选 的 ， 
默认 值 为 tme0， 即 如 果 没 有 给 出 时 间 戳 则 使 用 本 地 当前 时 间 。 

date0 函 数 的 参数 format 的 格式 化 选项 如 表 18.1 所 示 。 

表 18.1 参数 format 的 格式 化 选项 
参数 说 了 明 

a 小 写 的 上 午 和 下 午 值 ， 返 回 值 为 am 或 pm 
A 大 写 的 上 午 和 下 午 值 ， 返 回 值 为 AM 或 PM 
B Swatch Internet 标准 时 ， 返 回 值 为 000 一 999 
d 月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 为 01 一 31 
D 
F 


星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 为 Mon 一 Sun 
月 份 ， 完 整 的 文本 格式 ， 返 回 值 为 January 一 December 
小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 

G 小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 0 一 23 

h 小 时 ，12 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 01 一 12 

H 小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 00 一 23 
i 

I 

] 

| 


有 前 导 零 的 分 钟 数 ， 返 回 值 为 00 一 59 
判断 是 否 为 夏令 时 ， 如 果 是 夏令 时 返回 值 为 1， 否则 为 0 
月 份 中 的 第 几 天 ， 没 有 前 导 零 ， 返 回 值 为 1 一 31 
星期 数 ， 完 整 的 文本 格式 ， 返 回 值 为 Sunday 到 Saturday 
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续 表 
参数 说 明 

判断 是 否 为 半年 ， 如 果 是 半年 返回 值 为 1， 否 则 为 0 
m 数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 为 01 一 12 
M 3 个 字母 缩写 表示 的 月 份 ， 返 回 值 为 Jan 一 Dec 
n 数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 
0 与 格林 威 治 时 间 相差 的 小 时 数 ， 如 0200 
RFC 818 格式 的 日 期 ， 如 Thu，21 Dec 2000 16:01:07 +0200 
Ss 秒 数 ， 有 前 导 零 ， 返 回 值 为 00 一 59 
S 每 月 天 数 后 面 的 英文 后 级 ，2 个 字符 ， 如 st、nd、rd 或 者 也 ， 可 以 和 j 一 起 使 用 
t 指定 月 份 所 应 有 的 天 数 
T 本 机 所 在 的 时 区 
U 从 UNIX 纪元 (January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 
w 星期 中 的 第 几 天 ， 数 字 表示 ， 返 回 值 为 0~6 
WwW ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 
等 4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 

2 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 
2 年 份 中 的 第 几 天 ， 返 回 值 为 0~366 
时 差 偏 移 量 的 秒 数 。UTC 西边 的 时 区 偏 移 量 总 是 负 的 ，UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 返 回 值 为 


-43200 一 43200 


全 注意 这 里 有 效 的 时 间 蕉 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54 到 2038 年 1 月 19 日 
03:14:07 ( 此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 ). 在 Windows 系统 中 此 范围 限制 为 从 1970 年 
1 月 1 日 到 2038 年 1 月 19 日 


要 


[aE 要 应 用 发 散 的 思维 方式 , 在 考虑 编程 方法 时 就 不 能 被 date0 函 数 自 身 的 功能 所 限制 ，date0 函 数 


获取 的 时 间 不 只 可 以 用 来 记录 历史 中 的 某 一 时 刻 ， 而 且 还 可 以 将 其 应 用 到 其 他 的 地 方 ， 作 为 一 个 有 规律 


的 随机 数 。 
在 本 项 目 中 的 发 货 单 填 单 模块 中 就 是 应 用 date0 函 数 获取 当前 时 间 和 车 辆 ID 作为 发 货 单 的 编号 。 关 键 
代码 如 下 : 
<?php 
echo date("YmdHis"); // 获 取 系 统 的 当前 时 间 
echo Sljid; // 获 取 车 辆 的 ID 


J 


其 运行 结果 如 图 18.25 所 示 。 


局 


图 18.25 发 货 单 编号 生成 的 运行 结果 
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| 


[人 技巧 程序 开发 不 只 是 要 看 到 其 表面 的 功能 和 属性 ， 更 重要 的 是 将 某 些 功能 或 者 属性 有 机 地 结合 起 
来 ， 从 而 开发 出 更 有 意义 的 功能 ， 这 才 是 程序 员 所 要 追求 的 。 例 如 ， 在 学 习 字 符 串 中 的 str_ireplace0O 函 
数 时 ， 不 但 要 知道 该 函数 的 功能 是 实现 字符 串 的 替换 功能 ， 而 且 要 学 会 如 何 巧 妙 地 运用 其 实现 更 好 的 功 
能 。 当 换个 角度 去 考虑 这 个 函数 的 功能 时 ， 对 这 个 函数 的 参数 字符 囊 进行 颜色 改变 。 这 时 实现 的 功能 就 
不 只 是 简单 的 字符 串 替换 ， 而 且 对 字符 串 实现 了 描 红 的 功能 。 


18.7.3 ”发 货 单 填 单 的 实现 过 程 
发 货 单 的 填写 是 客户 在 确定 物流 配送 路 线 后 填写 的 一 个 货物 配送 详细 信息 单据 ， 内 容 包 括 发 货 单 编号 、 


车 牌号 码 、 车 主 电话 、 发 货 人 、 发 货 人 电话 、 发 货 地 址 、 付 款 方 式 、 收 货 人 、 收 货 人 电话 、 收 货 地 址 和 配 
送 说 明 。 发 货 单 填 单 的 运行 结果 如 图 18.26 所 示 。 


发 语音 六 了 3 1113251452013 于 于 号 码 : [re ED 
EE 2 人 | Hen: [用 司 
RR | 

加 
FE 
wwe 四 
加 
AR Pr ee 
站 下 
中 地址: 量 
—— 到 _ 
[3 FE| 
RN: 
上 到 


— 
图 18.26 发 货 单 填 单 的 运行 结果 


发 货 单 填 单 的 实现 通过 insert_dds.php 文件 来 完成 ， 首 先 随机 生成 一 个 订单 编号 , 然后 创建 一 个 form 表 
单 ， 最 后 将 表单 中 的 数据 提交 到 数据 库 中 。 在 insert_dds.php 文件 中 使 用 的 重要 表单 元 素 如 表 18.2 所 示 。 


表 18.2 发 货 单 填 单 中 使 用 的 重要 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 
Ne ee method="post” action="insert_dd ok.php" onSubmit="javascript: return 表单 
check ddO 
fahuo id text id="fahuo id" value="<?php echo date("YmdHis"):?><?php echo $ljid:?>”| 发 货 单 编号 
car number text value="<?php echo $myrow[car number]?>" 车 牌号 码 
car tel text value="<?php echo $myrow[tel]:?>" 车 主 电话 
和 a po ae 发 货 人 付款 <option> 付款 方式 
<option> 第 三 方 付 款 </option> 
Car lo; textarea Cols="65" rows="5" 说 明 


将 表单 中 提交 的 数据 存储 到 数据 库 是 通过 insert_dd_ok.php 文件 来 完成 的 , 在 该 文件 中 首先 连接 数据 库 ， 
然后 通过 preg_match0 函 数 判 断 表 单 中 提交 的 电话 号 码 的 格式 是 否 正确 ， 最 后 将 表单 中 提交 的 数据 存储 到 
tb_shopping 表 中 ， 并 且 将 车 辆 使 用 的 说 明 提 交 到 tb_car_log 表 中 ， 将 客户 信息 提交 到 tb_customer 表 中 。 其 


关键 代码 如 下 : 
@ 
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(代码 位 置 ， 光盘 \TM\Instance\18\insert_dd_ok.php) 


<?php 

session_start(); 

i($_SESSION['admin_user]==trueX{ // 淹 断 登录 的 用 户 是 否 正确 
include("conn/conn.php"); /| 连接 数据 库 
$fahuo_time=date("Y-m-d H:i:s"); 1/ 获取 系统 当前 时 间 


/eorexxerxerxemexs 关 断 表 单 中 提交 的 电话 号 码 的 格式 是 否 正确 reeeretrrsr oj 
if(preg_match("/*(\d{3})(\d{8))$I (a{4})(d{07})SI‘(d{4})(d{8})$I^(d{11})$/",$_POSTTcar_tel],S$countes)X 


if(preg_match("/*(\d{3}-)(\d{8}))$I‘(\d{4})(d{7})SI^(d{4}-)(d{8})sI^(d{11)5/",s_POST[fahuo_tel],Scountes)X{ 


if(preg_match("/^(\d{3}-)(\d{8})$I^(\d{4}-)(\d{7})$I^(d{4}-)(\d{8})$I^(d{11})$/",$_POSTT'shouhuo_tel, $countes)X{ 
wees 将 表单 提交 的 数据 添加 到 tb_shopping 表 中 * rerrrerreergr a 
Squery="insert into tb_shopping (car_number,car_tel,fahuo_id,fahuo_user, fahuo_tel,fahuo_address, 
fahuo_content,fahuo_time,fahuo_fk,shouhuo_user,shouhuo_address,shouhuo_tel)values(".$_POST[car_num 
ber]."".$_POST[car tel].",".$_POSTTfahuo id].",".$_POST[fahuo_user].",".$_POST[fahuo tel].",".$_PO 
ST[fahuo_address].",".$_POST[fahuo_content].",".$_POST[fahuo_time].",".$_POST[fahuo fk].",".$_POS 
T[shouhuo_user].",".$ POSTr'shouhuo_address'].",".$_POST['shouhuo_tel].")" 
$result=mysql_query($query); 
"erriexexntxxtexst 将 车 辆 使 用 说 明 添加 到 tb_car_log 表 中 * rssrrssrssrtsetswtws 4 
$querys="insert into tb_car_log(car_log,car_number,log_date,fahuo_id)values(".$_POST[car_log].",".$_POST 
[car_number].",".$_POST[fahuo time]","$_POST[fahuo_id].")" 
$results=mysql_query($querys); 
人 * xxzkhtttwtwtttttttttt 将 客户 信息 提交 到 tb_customer 表 中 * exwzsrrsrtswtwskwksets 由 
$queryss="insert into tp_customer (customer_user,customer_tel,customer_address)values(".$_POST[fahuo_user].”", 
".$_POST[fahuo_tel].",".$_POST[fahuo_address].")” 
$resultss=mysql_query($queryss); 
echo "<script>alert(' 发 货 单 添加 成 功 '");window.location.href='indexs.php?Imbs= 发 货 单 ';</script>"; 
Jelse{ 
echo "<script>alert(' 您 输入 的 收 货 人 的 电话 号 码 格式 不 正确 !);history.back()</script>"; } 
Jelse{ 


echo "<script>alert(' 您 输入 的 发 货 人 的 电话 号 码 格式 不 正确 !');history.back()</script>"; } 
}else{ 
echo "<script>alert(' 您 输入 的 车 主 的 电话 号 码 格式 不 正确 !');history.back()</script>"; } 


} 
Jelse{ // 判 断 用 户 是 否 正确 登录 ， 如 果 不 是 则 返回 登录 页 面 


echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>";} 
?> 


18.7.4 发 货 单 查询 的 实现 过 程 


发 货 单 查询 是 为 了 便于 对 发 货 单 进行 查找 以 en 珊 
及 处 理 而 设计 的 一 个 功能 ， 通 过 其 可 以 准确 地 查 。” | swss- nas rs 
找到 指定 的 发 货 单 ， 并 且 还 设置 一 个 发 货 单 删 除 -至 居 。 加 


的 功能 ， 可 以 对 已 经 失效 或 者 作废 的 发 货 单 进行 
删除 。 其 实现 的 原理 主要 是 通过 以 发 货 单 编号 为 “| 车 
条 件 的 精确 查找 或 者 以 发 货 人 姓名 为 条 件 的 模糊 ”| was FFF 
查询 ， 然 后 将 查询 的 结果 输出 到 页 面 中 。 运 行 结 。 Bm 
果 如 图 18.27 所 示 。 一 

发 货 单 查询 的 实现 主要 通过 hwys php 文件 完 
成 ， 首 先 与 数据 库 建立 连接 ， 然 后 创建 一 个 表单 和 


@ 


[7 
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fomm1， 通 过 该 表单 来 提交 查询 的 条 件 ， 可 以 选择 精确 查找 〈 以 发 货 单 编 号 为 条 件 ) 或 者 模糊 查询 (以 发 货 
人 姓名 为 条 件 ) ， 最 后 根据 fomml 表单 中 提交 的 数据 ,执行 查询 语句 ， 从 数据 库 中 读 取 出 符合 条 件 的 发 货 单 
的 内 容 。 程 序 的 关键 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\18\hwys.php) 


<?php session_start(); include("conn/conn.php"); // 初 始 化 session 变量 ， 连 接 数 据 库 
if($_SESSION[‘admin_user]==trueX{ // 判 断 管理 员 是 否 是 正确 登录 

?> 

<!--> 省 略 了 部 分 代码 <!--> 

<?php 


/* saxxxxxxx#xxxs 关 | 断 当 form1 表单 提交 的 值 为 发 货 单 编号 时 ， 执 行 下 面 的 语句 * xsxrxxtszxxxx xj 
放 $_POST[select1]==" 发 货 单 编号 "and $_POST[select1]!="X{ 

$query="select * from tb_shopping where fahuo_id=".$_POST[select1].”"'; 
Sresult=mysql_query($query); /| 执行 查询 语句 
$myrow=mysql_fetch_array($result); /获取 查 结果 


} 
weswrxerxwerwrst 如 果 form1 表单 提交 的 值 为 发 贷 人 了 时， 执行 下 面 的 语句 * areweseeresrrs */ 
if($_POST[select1"]==" 发 货 人 " and $_POST[select1]i="){ 
$query="select * from tb_shopping where fahuo_user=".$_POST['select1]."™"; 
$result=mysql_query($query); // 检 索 指定 发 货 人 的 信息 
$myrow=mysql_fetch_array($result); 
全 
… /省 略 部 分 HTML 代码 

<td colspan="4" bgcolor="#FFFFFF"><textarea name="car_log"> 

< 上 一 一 -输出 车 辆 日 志 中 的 车 辆 使 用 情况 一 一 > 

<?php 
$querys="select * from tb_car 二 where fahuo_id='$myrowlfahuo_id]"; 人 日 志 表 中 查找 指定 的 数据 
Sresults=mysql_query($querys); /执行 查询 语句 
$myrows=mysql_fetch_array($results); [下 | 
echo $myrows['car_log’]; /| 输出 车 辆 日 志 信息 
?> 


</textarea></td> 
Te 根据 发 货 单 的 ID 设置 一 个 发 货 单 删除 的 超 链接 一 一 一 一 一 -一 -一 一 -> 
<td bgcolor="#FFFFFF"><a href="fhd_qr.php?delete=<?php echo $myrow['id1];?>"> 删 除 发 货 单 </a></td> 
… // 省 略 部 分 HTML 代码 
<?php }else{ // 判 断 管理 员 是 否 正 确 登录 ， 如 果 错 误 则 返回 到 登录 页 
echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>"; 
} 


?> 
发 货 单 删除 的 操作 通过 fhd_qr.php 文件 来 完成 ， 主 要 根据 超 链接 中 提供 的 发 货 单 ID 执行 删除 的 操作 。 
其 关键 代码 如 下 : 
《代码 位 置 ， 光盘 \TM\Instance\18\fhd_qr.php) 


<?php session_start(); include("conn/conn.php"); /初始 化 session 变量 ， 连 接 数据 库 
if($_SESSION[admin_user]==trueX{ 
if$_GET[fahuo_id]==truejf // 判 断 提交 的 发 货 单 ID 是 否 为 真 


$query="delete from tb_car_log where fahuo_id=".$_GET[fahuo_id].”"'; 
// 根 据 发 货 单 ID 执行 删除 发 货 单 的 操作 
Sresult=mysql_query($query); 


$query="update tb ee Set fahuo_ys='1 where fahuo_id=".$_GET[fahuo_id].™" "; 

Sresult=mysql_query($query): 

echo "<script>alert(' 发 货 回 执 单 处 理 成 功 !):window; location.href='indexs.php?Imbs= 回 执 发 货 单 确认 
“</script>"; 


} 
if($_GET[ delete]==trueX{ 
$query="delete from tb_shopping where id=".$_GET[delete].”"; 
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$result=mysql_query($query); 


/* xxxtyaktrxttasttx 引 发 化 单 出 除 成 功 ， 弹 出 提示 ， 并 返回 到 上 一 页 * xsyxzxxsxrxstszsasx 
echo "<script>alert(' 发 货 单 删 除 成 功 !"");window.location.href='indexs.php?Imbs= 发 货 单 查询 ';</script>"; 


} 
jelse{ 


wereeerreerrserneerrsr 发 货 单 删除 失败 ， 弹 出 提示 ， 并 返回 到 上 一 页 * rerrerrrerrreerrerrrerr 
echo "<script>alert(' 请 您 正确 登录 "); window .location .href='index.php';</script>"; 


} 


?> 


18.7.5 发 货 单 打印 的 实现 过 程 


发 货 单 的 打印 是 为 货物 的 发 送 提供 一 个 书面 的 依据 ， 作 为 
企业 与 客户 商业 活动 的 凭证 ， 其 内 容 必须 真实 、 准 确 。 发 货 单 
打印 的 运行 结果 如 图 18.28 所 示 。 

这 里 的 发 货 单打 印 技术 主要 通过 框架 打印 技术 来 完成 ， 即 
只 打印 框架 中 的 内 容 ， 而 网 页 中 其 他 的 内 容 不 打印 。 实 现 原理 
是 : 首先 在 insert_dd.php 文件 中 创建 一 个 浮动 框架 ， 设 置 要 指 
定 打印 内 容 的 范围 ， 并 且 连 接 到 要 打印 的 insert_dds.php 文件 ， 
然后 在 指定 的 文件 中 输出 要 打印 的 内 容 , 最 后 应 用 onclick 事件 
调用 parent.content,focus0 和 window.print0 方 法 实现 打印 功能 。 

在 insert_dd.php 文件 中 创建 一 个 浮动 框架 ， 设 置 框架 名 称 
为 content， 设 置 打印 的 范围 宽 97%， 高 252%， 连 接 到 要 打印 
的 insert_dds.php 文件 。 程 序 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\18\insert_dd.php) 


ja 禄 证 中 【 疯 过 页 最 


[Mo ,| 


1 人口 


白雪 


[| 一 阳江 


图 18.28 发 货 单打 印 的 运行 结果 


<iframe name="content" src="insert_dds.php?ljid=<?php echo $ljid;?>" frameborder="0" width="97%" 


height="252%"></iframe> 


设置 要 打印 的 内 容 ， 通 过 insert_dds.php 文件 来 完成 ， 发 货 单 填写 的 内 容 以 表单 的 形式 显示 ， 其 中 的 内 


容 可 以 参考 光盘 中 TM\18\insert_dds.php 文件 ， 这 里 不 再 闭 述 。 


最 后 回 到 insert_dd.php 页 中 ， 在 该 页 中 设置 一 个 超 链 接 ， 应 用 onClick 事件 调用 parent.content.focusO 


和 window.print0 方 法 实现 发 货 单 的 打印 功能 。 程 序 关键 代码 如 下 : 


(代码 位 置 ， 光 盘 \TM\Instance\18\insert_dd.php) 
<a href="#" 
‘onClick="parent.content.focus();window.print();"> 


<img src="images/dl_033.gif" width="87" height="27" border="0"> 


</a> 


18.8 ”回执 单 验 收 管理 模块 设计 


车 4 视频 讲解 : 光盘 \TM\Video\ 第 18 章 \ 回 执 单 验收 管理 模块 设计 .exe 


18.8.1 回执 单 模块 概述 


el 


执 单 模块 的 主要 功能 就 是 对 货物 配送 完成 确认 。 将 该 发 货 单 执行 类 型 更 新 为 “1”， 表 明 该 次 物流 配 
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送 已 经 完成 清空 该 车 辆 的 使 用 日 志 ， 便 于 执行 下 一 个 订 和 
单 。 该 模块 的 业务 操作 流程 如 图 18.29 所 示 。 


18.8.2 MySQL 函数 库 函 数 应 用 技术 人 Ey 


清空 
在 回执 单 模块 中 ， 要 对 回执 单 进行 确认 首先 要 查找 到 。 


指定 的 发 货 单 ， 并 且 对 其 内 容 进行 核实 。 这 里 应 用 的 是 根 18.29 回执 单 模块 的 业务 操作 流程 
据 发 货 单 的 编号 进行 查询 ， 从 数据 库 中 读 取 出 符合 条 件 的 数据 。 在 执行 查询 语句 、 从 数据 库 中 读 取 数据 时 ， 
主要 应 用 mysql_ query0、mysql_num rows() 和 mysql_fetch_array0 函 数 。 

首先 通过 mysql_query0 函 数 执行 一 个 select 查询 语句 ， 然 后 通过 mysql_num rows() 函 数 判 断 查 询 的 结 
果 是 否 有 值 ， 如 果 没 有 值 则 输出 “您 查找 的 发 货 单 编号 不 存在 ! ”， 如 果 有 值 则 执行 mysql_fetch_arrau() 函 
数 ， 输 出 数据 库 中 查询 到 的 数据 。 

mysql_query0 函 数 向 与 指定 的 连接 标识 符 关 联 的 服务 器 的 当前 活动 数据 库 中 发 送 一 条 MySQL 查询 。 语 
法 如 下 : 

resource mysql_query ( string query [, resource link_identifier] ) 

mysql_query0 函 数 的 参数 query 是 字符 串 的 类 型 ， 传 入 的 是 SQL 指令 ; 参数 link identfier 是 资源 类 型 ， 
传 入 的 是 由 mysql_connect0 函 数 或 者 mysql_pconnect0 函 数 返回 的 连接 号 。 如 果 省 上 略 该 参数 ， 则 会 使 用 最 后 

-个 打开 的 MySQL 数据 库 连接 。 


人 


mysql_ num rowsO 函 数 获取 结果 集中 行 的 数目 。 语 法 如 下 : 
int mysql_num_rows ( resource result ) 


人 注意 该 函数 仅 对 SELECT 语句 有 效 。 
mysql_fetch_array0 函 数 返回 根据 从 结果 集 获 取 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false。 语 法 如 下 : 


array mysql_fetch_array ( resource result [, int result_type] ) 
mysql_fetch_array0 函 数 的 参数 说 明 如 表 18.3 所 示 。 
表 18.3 mysql_fetch_array() 函 数 的 参数 说 明 
参数 名 称 说 明 
result 资源 类 型 的 参数 ， 要 传 入 的 是 由 mysql_query0 函 数 返回 的 数据 指针 


可 选 参数 ,整数 型 参数 ， 要 传 入 的 是 MYSQL ASSOC、MYSQL NUM 和 MYSQL BOTH 3 种 由 PHP 
定义 好 的 常数 之 一 ， 默 认 值 是 MYSQL BOTH 
result type | 用 MYSQL ASSOC 只 得 到 关联 索引 (相当 于 mysql_fetch_assocO 函 数 ) 
用 MYSQL NUM 只 得 到 数字 索引 (相当 于 mysql 人 etch_row0 函 数 ) 
用 MYSQL BOTH 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 


和 0 注 站 本 函数 返回 的 字段 名 是 区 分 大 小 写 的 。 
通过 上 述 3 个 函数 就 可 以 完成 从 数据 库 中 读 取 数据 ， 并 且 将 数据 输出 到 页 面 的 功能 。 


全 
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pe ee i 
据 库 中 所 有 符合 条 件数 据 的 循环 输出 。 


应 用 while 循环 语句 读 取 数据 库 中 的 数据 ， 这 里 以 读 取 tb_shopping 表 中 的 数据 为 例 。 代 码 如 下 : 
<?php include("conn/conn.php"); // 连 接 数据 库 


$query="select * from tb_shopping "; /| 编写 查询 语句 
$result=mysql_query($query); /执行 查询 语句 

@ if(mysql_num_rows($result)<1}{ / 削 断 数据 库 中 是 否 存 在 数据 
echo "您 查找 的 内 容 不 存在 必 ; 
Jelse{ 

@ while($myrow=mysql_fetch_array($result)X{ // 循 环 输出 数据 库 中 的 数据 

er 

<?php 

echo $myrowfid]; // 输 出 数据 库 中 数据 

Ei 

四 // 输 出 内 容 的 代码 部 分 略 

<?php }?> 


oh, 
SC @ mysql num rows0: 返回 根据 从 结果 集 获取 的 行 生成 的 数组 ， 如 果 没 有 更 多 行 则 返回 false。 
@ mysql fetch array0: 获取 结果 集中 行 的 数目 。 该 函数 只 对 select 语句 有 效 。 


18.8.3 回执 单 验收 模块 的 实现 过 程 


回执 单 验收 模块 的 主要 功能 就 是 对 货物 配送 完成 确认 ， 运 行 
结果 如 图 18.30 所 示 。 

回执 单 验收 模块 的 实现 过 程 是 : 首先 在 select_dhd.php 页 中 根 
据 发 货 单 的 编号 查询 出 指定 的 发 货 单 ， 对 发 货 单 进行 核实 ， 然 后 
单 击 “ 发 货 订单 确认 ” 超 链接 ， 实 现 回执 单 的 验收 。 其 关键 代码 
如 下 


(代码 位 置 ， 光盘 \TMInstancelSselect_dhd.php) 图 18.30 回执 单 验收 模块 的 运行 结果 
<?php 
include("conn/conn.php"); /| 连接 数据 库 


if($_POST[Submit]==trueX{ 
$query="select * from tb_shopping where fahuo_id=".$_POST[select].""'; // 编 写 查 询 语句 
询 语 


$result=mysql_query($query); /| 执行 查询 语句 
if(mysql_num_rows($result)<1)}{ /| 淹 断 数据 库 中 是 否 存 在 数据 
echo "您 查找 的 发 货 单 编号 不 存在 !"; 
}else{ 
$myrow=mysql_fetch_array($result); // 输 出 数据 库 中 的 数据 
?> 
/输出 内 容 的 代码 部 分 略 


<td colspan="4" bgcolor="#FFFFFF"> 
<textarea name="car_log"> 
<?php 
xxxkxxsxxsxtxtt 根 据 提交 的 货物 编号 ， 查 询 车 辆 日 志 中 的 对 应 日 志 xzxxxsxxxxxxxx 
$querys="select * from tb_car_log where fahuo_id='$myrow[fahuo_id]"; 
Sresults=mysql_query($querys); /执行 查询 语句 
Smyrows=mysql_fetch_array($results); // 输 出 数据 库 中 的 数据 
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echo $myrows[car_log]; 
?> 


</textarea></td> 
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// 输 出 数据 库 中 的 数据 


// 输 出 内 容 的 代码 部 分 略 


< 该 , 执 行 发 货 单 的 确认 一 > 
<td bgcolor="#FFFFFF"><a href="fhd_qr.php?fahuo_id=<?php echo $myrow[lfahuo_id];?>"> 发 货 订 单 回 执 


确认 </a></td> 


最 后 在 fthd_qr.php 页 根据 超 链 接 提供 的 发 货 单 编号 更 新 发 货 单 类 型 为 <1”， 清空 对 应 车 辆 的 使 用 日 志 。 


其 程序 的 关键 代码 如 下 : 


(代码 位 置 ， 光盘 \TM\Instance\18\fhd_gqr.php) 


<?php session_start(); include("conn/conn.php"); 
if($_SESSION[‘admin_user]==trueX{ 
if($_GET[‘fahuo_id]==trueX{ 


/初始 化 session 变量 
/判断 用 户 是 否 是 正确 登录 
// 判 断 超 链接 传递 的 变量 值 是 否 为 真 


resxwewxwtwswtsts 愉 超 链接 传递 的 变量 值 为 条 件 ， 编 写 删除 语句 **ssesreesrexxxey 岂 
$query="delete from tb_car_log where fahuo_id=".$_GET[fahuo_id].""; 


Sresult=mysql_query($query); 


/| 执行 删除 语句 
/zwxwtwxtwxwtwtwttxty 更 新 发 货 单 数据 表 中 发 货 的 类 型 为 “1?” wererewestwrerwtstettt 训 
$query="update tb_shopping set fahuo_ys='1 where fahuo_id=".$_GET[fahuo_id]."" 


$result=mysql_query($query); 


echo "<script>alert(' 发 货 回执 单 处 理 成 功 ");window ,location.href='indexs.php?Imbs= 回 执 发 货 单 


确认 ';</script>"; } 
Jelse{ 


echo "<script>alert(' 请 您 正确 登录 !"); window.location.href='index.php';</script>"; 


?> 


18.9 基础 信息 


管理 模块 设计 


名 1 视频 讲解 : 光盘 \TM\Video\ 第 18 章 \ 基 础 信息 管理 模块 设计 .exe 


18.9.1 基础 信息 管理 模块 概述 
基础 信息 管理 模块 主要 包括 客户 信息 管理 , 用 于 
添加 客户 信息 和 删除 客户 信息 ; 车 源 信息 管理 ， 主 要 
用 于 对 和 车 源 信息 进行 管理 ， 实 现 车 源 信息 的 添加 、 修 
改 和 删除 功能 ; 管理 员 信息 管理 ， 用 于 修改 管理 员 的 
密码 基础 信息 管理 模块 的 功能 结构 如 图 18.31 所 示 。 


18.9.2 面向 对 象 封 装 密码 修改 类 


修改 管理 员 密 码 的 模块 中 , 在 对 密码 进行 处 理 的 
过 程 中 用 的 是 面向 对 象 技术 实现 密码 的 修改 。 

面向 对 象 技术 的 关键 就 是 类 的 创建 , 首先 创建 一 
个 chkuser 类 ， 其 中 定义 $ 个 数据 成 员 ， 实 现 对 管理 
员 密 码 的 修改 ， 分 别 是 管理 员 名 称 ($admin_user) 、 


管理 员 


18.31 基础 信息 管理 模块 的 功能 结构 图 


_ 国 ) 
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管理 员 密码 ($admin pass) 、 管 理 员 新 密码 ($admin_new_pass) 、 管 理 员 新 密码 确认 ($admin new_pass2) 
和 验证 码 ($yzm) ， 然 后 应 用 chkuser 函数 实现 数据 成 员 的 初始 化 ， 最 后 应 用 数据 成 员 函 数 chkinput 实现 密 
码 的 修改 。 程 序 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\18\update_pass_ok.php) 


<?php session_start(); /初始 化 session 变量 

@ casschkuser{ /创建 一 个 chkuser 类 
var $admin_user; /创建 数据 成 员 ， 用 户 名 
var $admin_pass; /创建 数据 成 员 ， 密 码 
var $admin_new_pass; /创建 数据 成 员 ， 新 密码 
var $admin_new_pass2; // 创 建 数据 成 员 ， eel 
var $yzm; /创建 数据 成 员 ， 验 证 

function chkuser($x,$y,$m,$n,$zjf /创建 chkuser 函数 ， re 


Sthis->admin_user=$x; 

Sthis->admin_pass=$y; 
Sthis->admin_new_pass=$m; 
Sthis->admin_new_pass2=Sn; 
Sthis->yzm=$z; 


} 
@ function chkinput(){ // 创 建 chkinput 函数 实现 密码 更 改 
// 判 断 输 入 的 验证 码 是 否 正确 
© if(strval($this->yzm)!=$_SESSION["autonum1"]X{ 
echo "<script>alert( 验 证 码 输入 错误 !");history.back();</script>"; 
exit; 


include("conn/conn.php"); /| 连接 数据 库 
/判断 输入 的 密码 是 否 正确 
$query=mysql_query("select * from tb_admin where admin_user=".$this->admin_user." and admin_ 
pass=".$this->admin_pass."™"); 
if(mysql_num_rows($query)<1X{ 
nts 加 果 密码 不 正确 则 提示 重新 输入 密码 essaxsxs */ 
echo "<script>alert(' 您 输入 的 密码 不 正确 ， 请 重新 输入 ');history.back();</script>"; 
}else{ 
rn wa R 害 码 正确 ， 则 应 用 更 新 语句 ， 实现 密码 的 更 改 *************** wh 
$query1=mysql_query("update tb_admin set admin_pass=".$this->admin_new_pass.""); 
if($query1==true){ 
echo "<script>alert(' 密 码 更 改 成 功 ');history.back();</script>"; 
有 


revsereoeseoeeroeesereseeeoerrss 类 的 实例 化 过 程 Serrsrrrereserroseoseerserrseerser o 


@ $chk=new 
chkuser($_POST[admin_user],md5($_POST[admin_pass]),md5($_POST[admin_new_pass]),$_POST[admin_ 
new_pass2],$_POST[yzm]); 

$chk->chkinput(); 
?> 


SC 全 
@ chkuser: 创建 一 个 类 chkuser， 用 于 修改 管理 员 密 码 。 
@ chkinput0: 创建 一 个 自 定义 函数 chkinputO0， 用 于 实现 密码 修改 。 
日 strval0: 获取 变量 的 字符 串 。 
@ new chkuser: 实现 类 的 实例 化 ， 应 用 类 实现 密码 修改 。 


@® 
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18.9.3 ”客户 信息 管理 的 实现 过 程 


客户 信息 管理 模块 的 主要 功能 就 是 向 数据 库 中 添加 客户 信息 ， 并 且 可 以 对 指定 的 客户 进行 删除 操作 。 
客户 信息 管理 模块 的 的 运行 结果 如 图 18.32 所 示 。 
人 EL 
| 
下 站 生理 
ES FS 呈 RR [3 
和 ES EE 四 
1 Fe Er EG [2 


图 18.32 客户 信息 管理 模块 的 运行 结果 


客户 信息 管理 模块 主要 通过 customer.php、customer_ok.php 和 delete_customer.php 文件 来 完成 。 首 先 通 
过 customer.php 文件 来 添加 客户 信息 ， 并 且 输 出 数据 库 中 的 所 有 客户 信息 ,设置 删除 客户 信息 的 超 链接 。 其 
输出 客户 详细 信息 的 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\18\customer.php) 


<?php include("conn/conn.php"); /| 连接 数据 库 
$query=mysql_query("select * from tb_customer "); /执行 查询 语句 ， 从 数据 库 中 读 取 数 据 
while($myrow=mysql_fetch_array($query)}{ // 通 过 while 循环 语句 ， 循 环 输出 数据 库 中 的 数据 
?> 
<tr> 


<td height="27" align="center" bgcolor="#FFFFFF"><?php echo $myrow['customer_user"];?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo $myrow['customer_tel];?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo $myrow['customer_address'];?></td> 
< 上 一 一 一 一 一 设置 删除 客户 信息 的 超 链接 ， 并 且 设 置 客户 的 ID 为 变量 一 一 一 一 一 一 一 一 
<td align="center" bgcolor="#FFFFFF"><a href="delete_customer.php?delete=<?php echo 

$myrow['customer_id"];?>"> 删 除 </a></td> 

</tr> 

<?php }?> 
客户 信息 添加 的 操作 通过 customer_ok php 文件 完成 ， 首 先 连 接 数据 库 ， 然 后 判断 提交 的 电话 号 码 格式 
是 否 正确 ， 最 后 将 客户 信息 数据 提交 到 数据 库 中 。 程 序 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\18\customer_ok.php》 

<?php include("conn/conn.php"); /| 连接 数据 库 

if($_POST[Submit]==true and $_POST[customer_user]==true) { // 浏 断 提交 的 值 是 否 为 真 
aastt 由 下 于 于 下 下 下 下 下 粒 浊 | 睛 提交 的 电话 号 码 格式 是 否 正确 **xzsxxtststsxatsxayaxxyats 


if(preg_match("/*(\d{3})(\d{8})SI(\d{4}-)(d{7})SIANd{4}-)(d{8})sI^(d{11})$/",$_POST[customer_tel],$cou 


ntes)X{ 
产 “eeoewwewsoesgessoeog0s 梅 提交 的 数据 添加 到 客户 信息 表 中 **weeeweeeeeeseeesesesss wi 
Squery=mysql_query("insert into tb_customer(customer_user,customer_tel,customer_address) 
values(".$_POST[customer_user].",".$_POST[customer_tel].",".$_POST[customer_address].")"); 
if($query==trueX{ // 如 果 添 加 操作 成 功 ， 弹 出 提示 ， 并 返回 首页 
echo "<script> alert(' 客 户 信息 添 加 成 功 ');window.location.href='indexs.php?Imbs= 客 户 信息 
管理 ';</script>"; 
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} 
Jelse{ // 如 果 添 加 操作 失败 ， 弹 出 提示 ， 并 返回 上 一 页 
echo "<script>alert( 您 输入 的 电话 号 码 格式 不 正确 !");history.back()</script>"; 
’ 
器 
?> 


客户 信息 删除 的 操作 通过 delete_customer.php 文件 来 完成 ,主要 以 超 链 接 中 提交 的 变量 为 根据 ， 删 除数 
据 库 中 指定 的 客户 信息 。 程 序 的 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\18\delete_customer.php) 
<?php include("conn/conn.php"); /| 连接 数据 库 
if($_GETTdelete]==trueX{( // 判 断 提交 的 变量 值 是 否 为 真 
/* eastwwttwtstswtttt 根 据 $_GET 获取 的 变量 值 为 依据 ， 执 行 删除 语句 esszssrssryswysstsx 
$query=mysql_query("delete from tb_customer where customer_id=".$_GET[delete]."”); 
if($queryX{ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 ， 并 返回 首页 
echo "<script> alert(' 客 户 信息 添加 成 功 ');window.location.href='indexs.php?Imbs= 客 户 信息 管理 
"</script>"; 
} 
3 


?> 
18.9.4 车 源 信 息 管理 的 实现 过 程 


车 源 信息 管理 模块 主要 实现 对 车 辆 信息 的 添加 、 修 改 和 删除 操作 。 车 源 信 息 管 理 模块 的 运行 结果 如 
图 18.33 所 示 。 
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18.33 ”车 源 信息 管理 模块 的 运行 结果 
车 源 信息 管理 模块 主要 由 两 个 文件 组 成 。 一 个 是 car.php 文件 ， 以 表单 文件 为 主 用 于 输出 车 辆 的 信息 和 
提交 车 辆 的 信息 。 设 置 form2 表单 ， 根 据 车 牌号 码 从 数据 库 中 读 取 对 应 车 辆 的 信息 ， 进 而 实现 对 该 车 辆 信息 
的 修改 或 者 删除 操作 ;设置 forml 表单 ,用 于 显示 从 数据 库 中 读 取 的 车 辆 信息 和 直接 添加 车 辆 信息 到 数据 库 。 
forml 表单 中 使 用 的 重要 元 素 如 表 18.4 所 示 。 
表 18.4 ”车 源 信息 管理 模块 中 使 用 的 重要 表单 元 素 
名 称 | 元 素 类 型 重要 属性 含 义 


表单 


method="post" action="car_insert_ok.php" 
onSubmit="javascript:retum check car(form1):" 


forml form 


username text id="usermame" value="<?php echo $myrows[username]:?>" 车 主 姓 名 


党 
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续 表 
名 称 元 素 类 型 重要 属性 含义 
car number text id="car_ number" value="<?php echo $myrows[car number]:?>" 车 牌号 码 
car tel text value="<?php echo $myrow[tel]:?>" 车 主 电话 
User_ number text id="user number" value="<?php echo $myrows[user number]:?>" 身份 证 号 码 


car content textarea <?php echo $myrows[car_content]:?> 车 辆 描述 
address textarea <?php echo Smyrows[address]:?> 家 庭 地 址 
car road textarea <?php echo $myrows[car ro0ad]:?> 运输 路 线 
Submit submit Value=" 提 交 " 提交 表单 
Submit2 submit value=" 修 改 " 提交 表单 
Submit4 submit Value=" 删 除 " 提交 表单 


另 一 个 是 car_insert_ok.php 文件 ， 通 过 该 文件 实现 对 表单 中 提交 的 数据 进行 处 理 。 首 先 连接 数据 库 ， 然 
后 根据 表单 中 提交 的 值 进 行 判断 ， 当 按钮 SSubmit 的 值 为 “提交 ”时 , 将 提交 的 数据 存储 到 指定 的 数据 表 中 ; 
当 按钮 SSubmit2 的 值 为 “修改 ”时 ， 则 更 新 语句 ， 对 车 辆 的 信息 进行 更 新 ， 当 按钮 SSubmit4 的 值 为 “删除 ” 
时 ， 则 执行 删除 语句 ， 将 指定 的 车 辆 信息 删除 。 程 序 关键 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\18\car_insert_ok.php) 


<?php session_start(); include("conn/conn.php"); /初始 化 session 变量 ， 连 接 数 据 库 
@ if($_SESSION[admin_user]==trueX{ // 判 断 管理 员 是 否 是 正常 登录 
if($_POST[Submit==trueX{ // 判 断 表单 提交 的 值 是 否 为 "提交 " 
* wxexxexwtswreswtty 淹 电表 单 中 提交 的 电话 号 码 的 格式 是 否 正确 ?syyeyeyyererwyyewy oj 
2 ifpreg_match("/d{17}[\d|X]lvd(15y",$_POST[user_number],$counts)){ 


WO eA ta el TC OL DS $_POST[user_tel],$countes)){ 
/eeemsremrzarxaxxxx#a* 将 表单 中 的 数据 添加 到 tb_car 数据 表 中 **sxyxsvexxxxxsrxsxy o 
$query=mysql_query("insert into tb_car(username,user_number,tel,address,car_number,car_road, 
car_content)values(".$_POST[username'].",".$_POST['user_number].",".$_POST[user_tel].",".$_POST[ad 
dress].",".$_POST['car_number].",".$_POST['car_road'].",".$_POST[car_content].")"); 
if($query==true){ 
echo "<script>alert(' 车 源 信息 添加 成 功 !");window.location.href='indexs.php?Imbs= 车 
源 信息 管理 ';</script>"; } 
jelsef 
echo "<script>alert(' 您 输入 的 电话 号 码 格式 不 正确 !!);history.back()</script>"; } 
}else{ 
echo "<script>alert(' 您 输入 的 身份 证 号 码 的 格式 不 正确 !);history.back()</script>"; } 


} 
xysxysxAxtAthhtyhsxhsysts 和 tt 江 | 轩 [表单 提交 的 值 是 否 为 "修改 "strxsrxxwxxwsrxsywtyatk 训 
© if($_POST[Submit21]==" 修 改 "){ 
re Dairy Oe 让 
if(preg_match("\d{17}NdIXINd{15¥",$_POST[user_number'],$counts)}X{ 


if(preg_match("/*(\d{3})(\d{8}))$I*(\d{4}-)(d{7})SI Nd{4}-)(d{8})sI*(d{11})s/",S_POSTIuser_tel],Scountes)) 


/* xxexxwxwxwxsxwxtywyeywyeyty 更 居 tb_car 数据 表 中 的 数据 **exexexyxxxyxrxrxxyryrywywy oj 

$query="update th_car set usemame=".$_POST[usemame]", user_number= ".$_POST[user_ 
number].", tel=".$_POST[user tel].", address=".$_POST[address].", car_number= ".$_POST[car_number].", 
car_road=".$_POST[car_road]."， car_content=".$_POST['car_content]." where car_number=".$_POST 
['car_number].™; 

Sresult=mysql_query($query); 

if($result==true}{ 
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echo "<script>alert(' 车 源 数据 更 新 成 功 册 );:window .location.href='indexs.php?lImbs= 车 
源 信息 管理 ';</script>"; 
1 
jelsef 
echo "<script>alert(' 您 输入 的 电话 号 码 格式 不 正确 !!);history.back()</script>"; 
} 


Jelse{ 
echo "<script>alert(' 您 输入 的 身份 证 号 码 的 格式 不 正确 !);history.back()</script>"; 
外 


enn) 单 提交 的 值 是 否 为 "删除 
9 if($_POST[Submit4]==" 删 除 "{ 
Squery="delete from tb_car where car_number=".$_POST['car_number]."™; 
Sresult=mysql_query($query); 
if($result==true}{ 
echo "<script>alert(' 车 源 数 据 删 除 成 功 !");window.location.href='indexs.php?Imbs= 车 源 信息 管理 
“</script>"; 


} 
}else{ 

echo "<script>alert(' 请 您 正确 登录 ! '); window.location.href='index.php';</script>"; 
} 


?> 


of. 
SS @$_SESSION[admin_user]: 判断 用 户 是 否 是 正常 登录 。 

@ preg_match(): 应 用 preg_ matchO 验 证 电话 号 码 的 格式 是 否 正确 。preg _match() 对 正则 表达 式 进行 
匹配 ,返回 匹配 的 次 数 , 0 次 (没有 匹配 ) 或 者 1 次 ， 因 为 preg_matchO 函 数 在 第 一 次 匹配 后 将 停止 搜索 ， 
出 错 返回 false。 

@ $Submit2==" 修 改 ": 判断 表单 提交 的 值 是 否 等 于 修改 ， 如 果 等 于 修改 则 执行 修改 的 操作 。 

@ $Submit4 一 "删除 ": 判断 表单 提交 的 值 是 否 等 于 删除 ， 如 果 等 于 删除 则 执行 删除 语句 。 


18.10 小 结 


本 章 从 某 物流 配送 公司 的 实际 需求 角度 出 发 ， 开 发 一 个 完整 的 物流 配送 信息 管理 系统 ， 详 细 地 讲解 了 
物流 管理 系统 的 开发 流程 ， 从 最 初 的 需求 分 析 、 可 行 性 分 析 ， 到 系统 的 设计 、 数 据 库 的 设计 ， 其 中 重点 突 
出 车 源 信息 查询 模块 、 发 货 单 管理 模块 和 回执 单 验收 管理 模块 的 设计 。 通 过 本 章 的 学 习 ， 读 者 不 但 可 以 了 
解 物流 配送 系统 开发 的 整体 思路 ， 而 且 还 能 够 掌握 很 多 关键 的 技术 和 技巧 : 函数 的 灵活 运用 、MySQL 存储 
过 程 的 创建 和 应 用 、 报 表 打印 技术 的 实现 、 应 用 正则 表达 式 验 证 电话 号 码 等 。 
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ADODB 类 库 
(人 视频 讲解 : 64 分钟 ) 


ADODB 提供 了 一 套 完 整 的 方法 和 属性 让 开发 者 控制 数据 库 系 
统 。 在 操作 过 程 中 ， 无 须 计 较 使 用 的 是 什么 数据 库 ， 而 只 要 记得 它 的 
功能 即 可 。 因 为 在 更 换 数 据 库 时 ， 只 要 修改 一 个 属性 值 ，ADODB 就 
会 自动 依据 设 定 取 用 正确 的 PHP 函 教 。 此 外 ， 有 可 能 还 需要 修改 SQL 
语句 ， 因 为 不 同 的 数据 库 在 SQL 语句 的 使 用 上 有 一 点 区 别 。 这 就 是 
ADODB 的 魅力 所 在 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

mm 了 解 ADODB 类 库 是 什么 

m 了解 ADODB 支持 的 数据 库 

WH 熟悉 ADODB 类 库 的 安装 

mm 熟悉 ADODB 类 库 的 常用 方法 

Wl 熟悉 ADODB 类 库 的 应 用 

Mi 掌握 如 何 灵 活 运行 ADODB 类 库 


第 19 章 ADODB 类 库 


19.1 ADODB 类 库 是 什么 


ADODB (Active Data Objects Data Base) 是 存 取 数 据 库 所 使 用 的 一 组 函数 。 虽 然 PHP 是 构建 Web 系统 
强 有 力 的 工具 ， 但 是 由 于 PHP 存 取 函 数 一 直 没有 标准 化 ， 不 同 数据 库 间 的 函数 名 称 、 参 数 差异 很 大 ， 在 更 
换 数 据 库 时 ， 会 带 来 大 量 的 代码 修复 工作 。 这 时 ， 就 需要 一 组 函数 库 来 隐藏 不 同 数据 库 函数 间 的 差异 ， 让 


开发 者 可 以 很 简单 地 切换 数据 库 ， 这 就 是 ADODB 类 库 。 
ADODB 类 库 有 以 下 优点 : 


回 ”安装 简单 ， 易 学 易 用 。 提 供 了 与 微软 的 ADODB 相似 的 语法 功能 。 无 论 是 初学 者 ， 还 是 从 其 他 语 


言 转 过 来 的 开发 人 员 (主要 是 ASP) ， 都 可 以 很 快 上 手 。 


以 标准 的 SQL 语句 书写 的 程序 在 更 换 数据 库 时 ， 不 需要 改变 源 程序 。 


回 
可 以 生成 Smarty 循环 需要 的 二 维 数组 ， 简 化 了 Smarty 开发 。 


支持 缓存 查询 ， 最 大 可 能 地 提高 查询 速度 。 


ADODB 类 库 的 缺点 是 执行 效率 慢 。 由 于 ADODB 类 库 非常 庞大 ， 仅 是 主 执行 类 (adodb.incphp) 就 有 


125KB， 所 以 在 使 用 ADODB 类 库 时 要 充分 考虑 到 这 一 点 。 


19.2 ADODB 支持 的 数据 库 


ADODB 目前 支持 MySQL、Oracle、Microsoft SQL Server、Sybase、PostgreSQL、FoxPro、Access、ADO 
和 ODBC 等 大 多 数 数据 库 。ADODB 所 支持 的 常见 数据 库 及 参数 如 表 19.1 所 示 。 


表 19.1 ADODB 支持 的 数据 库 及 参数 


测试 状态 资料 库 RecordCount() 支 持 


Microsoft Access/Jet, = 
ODBC/DSN 


数据 库 名 称 


access 


DB2， 可 以 通过 ODBC 获得 可 以 信赖 
的 运作 效果 
Microsoft Visual FoxPro， 需 要 建立 一 


需要 安装 的 驱动 


ODBC 


DB2 CLLODBC interface 


Windows 


UNIX 
Windows, 


vp 个 ODBC/DSN ODBC Windows 
> UNIX 
mssql 在 日 期 格式 上 仍 有 一 些 问 题 Mssql client 
Windows 
UNIX 
1 MySQL 理 MySQL client 
mysql ySQL 支持 交易 处 ySQL client i 


zs 或 MySQL 支持 交易 处 理 YN MySQL client 
maxsql Windows 
Oracle 8/9, 支持 比 Oracle 驱动 程式 还 
oci8 多 的 功能 。 在 连接 之 前 ， 可 能 需要 先 |Y/N Oracle client 
Windows 


配 好 环境 变数 (ORACLE HOME=...) 
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续 表 
数据 库 名 称 | 测试 状态 资料 库 RecordCount() 支 持 需要 安装 的 驱动 操作 系统 
UNIX 
oci8po Oracle 8/9 可 携 式 驱动 程式 YN Oracle client 
Windows 
db 标 必 DB ? ds on database |ODBC 
ok 用 PConnect(DSN'user.pwd) .连接 | eps on Windows, 
UNIX 
odbc_mssql 用 ODBC 连接 MSSQL ODBC 
Windows 
odbc_oracle 用 ODBC 连接 ORACLE ODBC 和 
Windows 
UNIX 
postgres PostgreSQL 不 支援 LIMIT 指令 PostgreSQL client 
Windows 
UNXI 
b: Syb: Sybase client 
sybase ybase ybase clienmr eg 


表 19.1 说 明 如 下 : 
A 表示 已 经 经 过 很 多 验证 及 测试 ， 可 靠 度 最 高 。 


加 加 加 加 


B 表示 已 经 测试 并 使 用 ， 但 可 能 仍 有 一 些 功 能 没有 达成 。 

C 表示 使 用 者 自行 配置 或 试用 的 驱动 程式 ， 可 能 没有 完全 支持 ADODB 的 功能 。 
“RecordCount0 支 持 ” 是 指 RecordCountO 函 数 是 否 会 回 传 用 SELECT 指令 取得 的 记录 数 〈 不 支持 
传 回 -1) 。Y/N 表示 当 全 局 变量 SADODB_COUNTER=true 时 ， 会 以 模拟 的 方式 取得 ， 但 如 果 估 计 
到 记录 数 会 很 大 时 ， 最 好 把 这 个 值 设 为 false， 也 就 是 关 掉 模拟 功能 ， 否 则 会 耗 掉 很 多 内 存 。 该 变 
量 在 每 次 执行 时 都 会 检查 ， 所 以 可 以 选择 性 地 使 用 。 


19.3 ADODB 下 载 与 安装 


要 使 用 ADODB 来 操作 数据 库 ， 首先 就 要 获取 和 安装 ADODB。 读者 只 要 到 网 上 下 载 ADODB 包 , 解压 


到 Web 服务 器 目录 下 即 可 。 本 书 使 用 的 ADODB 的 版 本 是 5.0。 


人 注意 要 使 用 ADODB， 使 用 的 PHP 必须 是 4.0.1 以 上 的 版 本 。 
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在 详细 介绍 ADODB 类 库 之 前 ， 首 先 了 解 一 下 使 用 ADODB 类 库 的 简要 步骤 。 

例 19.1 测试 ADODB 是 否 配置 成 功 。 应 用 ADODB 类 库 连 接 MySQL 数据 库 ， 指 定 连接 的 数据 库 为 

db_database19， 使 用 的 数据 表 是 tb_object。 将 ADODB 类 库存 储 在 根 目录 的 上 级 目录 中 ， 名 称 为 adodb5。 
(实例 位 置 ， 光盘 \TM\Instance\19\19.1) 


代码 如 下 : 
<?php 
include_once (..Jadodb5/adodb.inc.php'); 


// 载 入 adodb.ini.php 文 件 


$conn = ADONewConnection('mysql"); /连接 MySQL 数据 库 
$conn -> PConnect(localhost,root,"111',"db_database19"); /连接 db_database19 库 
$conn -> execute('set names gb2312"); ll 设 置 编码 

Srst = $conn -> Execute("select * from tb_object") or die( 执 行 错误 '); /执行 SQL 语句 


while(!$rst -> EOFX{ 


// 配 合 while 语句 循环 输出 结果 
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echo $rst -> fields[1].'<br>"; 
Srst -> movenext(); /指针 下 移 


Srst -> close(); 


$conn -> close(); 
?> 


结果 为 : 图 书 开发 网 络 电器 。 

使 用 ADODB 类 库 的 一 般 操 作 步 骤 为 : 

(1) 载 入 adodb.inc.php 文件 。 要 使 用 ADODB 类 库 ， 首 先 要 启动 ADODB， 启 动 ADODB 的 方法 就 是 
载 入 adodb.iniphp 文件 。 

(2) 建立 连接 。 使 用 ADONewConnection0 函 数 。 

(3) 连接 数据 库 。 使 用 PConnectO 函 数 。 

(4) 执行 数据 库 操作 。 使 用 Execute0 方 法 返回 结果 集 。 

(5) 对 结果 集 进行 操作 。 

(6) 关闭 连接 。 


19.4 ADODB 类 库 


ADODB 类 库 非 常 庞大 ， 仅 一 个 adodb.inc.php 文件 就 有 120KB， 所 以 这 里 只 介绍 一 些 常 用 函数 和 变 
量 。 这 些 函 数 按照 功能 的 不 同 可 分 为 连接 数据 库 函数 、 操 作 数据 库 函 数 、 结 果 集 处 理 函 数 和 错误 处 理 函 
数 等 。 


19.4.1 连接 数据 库 函 数 


区 视频 讲解 : 光盘 \TMIVVideo\ 第 19 章 \ 连 接 教 据 库 函数 .exe 

在 介绍 连接 数据 库 的 方法 之 前 ， 先 来 了 解 一 些 在 连接 数据 库 过 程 中 应 用 到 的 函数 。 

ADONewConnection($databaseType) 

函数 作用 : 连接 数据 库 系统 。 

参数 $databaseType 表示 要 连接 的 数据 库 系统 的 名 称 ， 如 mysql、mssql 等 。 

PConnect($host,[$user],[$password],[$database]) 

函数 作用 : 持久 化 连接 数据 库 。 

参数 说 明 如 下 。 

> ”$host: 数据 库 系 统 的 服务 器 所 在 地 址 。 如 果 是 本 机 操作 ， 格 式 为 localhost。 
> $user: 数据 库 用 户 名 。 

> $password: 数据 库 密码 。 

> $database: 使 用 到 的 数据 库 。 

Connect0 也 是 连接 数据 库 的 函数 。 该 函数 为 非 持久 化 连接 数据 库 。 持 久 化 连接 和 非 持久 化 连接 的 区 别 
在 于 : 持久 化 连接 不 用 每 次 都 创建 新 连接 ， 可 以 提高 程序 的 执行 速度 。 但 有 些 数据 库 不 支持 ， 如 果 遇 到 不 
支持 的 数据 库 ， 可 以 使 用 Connect0 函 数 代 蔡 PConnect0 函 数 。 

ADODB 支持 多 种 数据 库 的 连接 ， 下 面 简 单 介绍 几 种 常用 数据 库 的 连接 方法 。 


图 


PHP+MySQL 开发 实战 


1. MySQL 
连接 MySQL 数据 库 有 两 种 不 同 的 方法 。 
第 一 种 使 用 标准 的 连接 字符 串 来 连接 。 


代码 如 下 : 

<?php 
include_once ('../adodb5/adodb.inc.php’); / 载 入 (include) adodb.inc.php 文 件 
S$conn = ADONewConnection('mysql'); /建立 连接 
$conn -> PConnect('localhost','root','111','db_database19"); /连接 数据 库 

让 

其 中 ，localhost、root、111 和 db_database19 分 别 表示 要 连接 数据 库 服务 器 的 地 址 、 用 户 名 、 密 码 和 数 

据 库 名 称 。 

第 二 种 是 通过 数据 源 名 称 (DSN) 的 方式 进行 连接 。 

代码 如 下 : 

<?php 
include_once ('../adodb5/adodb.inc.php'); // 载 入 (include) adodb.inc.php 文 件 
$conn = ADONewConnection('mysql://root:111@localhost/db_database19'); /连接 数据 库 

?> 

其 实现 的 功能 是 相同 的 。 


2. Microsoft SQL Server 
连接 Microsoft SQL Server 数据 库 使 用 ODBC 的 连接 方法 。 


代码 如 下 : 

<?php 
include_once ('../adodb5/adodb.inc.php'); // 载 入 (include) adodb.inc.php 文 件 
S$conn = ADONewConnection('odbc_mssql'); /| 连接 SQL 数据 库 
$conn-> PConnect("Driver={SQL Server};Server=localhost;Database=mydb;",'username','password'); 

?> 

其 中 ，localhost、username、password 和 mydb 分 别 表示 要 连接 数据 库 服务 器 的 地 址 、 用 户 名 、 密 码 和 

数据 库 名 称 。 


例 19.2 通过 ADODB 使 用 ODBC 方法 连接 Microsoft SQL Server 数据 库 ， 查 询 Microsoft SQL Server 
系统 库 master 中 的 表 systypes, 查询 条 件 是 xtype 的 值 为 173, 并 输出 查询 结果 .( 实 例 位 置 :光盘 \TM\Instance\ 


19\19.2) 
代码 如 下 : 
<?php 
include_once (../adodb5/adodb.inc.php'); /1 载 入 (include) adodb.inc.php 文 件 
S$conn = ADONewConnection(odbc_mssql'); /| 连接 SQL 数据 库 
$conn-> PConnect("Driver={SQL Server};Server=MR-NXT\NXT;Database=master;",'sa',"); 
$ADODB_FETCH_MODE = ADODB_FETCH_ASSOC.; /设置 字段 为 结果 集 索引 
$sqlstr = 'select * from systypes where xtype = ?'; /| 创建 SQL 语句 
Srst = $conn -> execute($sqlstr,'173') or die('connect error); 。” // 执 行 SQL， 查 询 xtype 为 173 的 数据 
while(l$rst -> EOF){ // 如 果 没 有 错误 , 则 配合 while 循环 输出 结果 
echo $rst -> fields['name'].'=>".$rst -> fields[xtype]."; /| 输出 查询 结果 
Srst -> movenext(); /指针 下 移 
} 
Srst -> close(); /关闭 连接 


$conn -> close(); 
?> 


结果 为 : binary=>173。 


局 
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3. Microsoft Access 
连接 Access 数据 库 使 用 的 是 ODBC 方法 。 


代码 如 下 : 

<?php 
include_once ('../adodb5/adodb.inc.php’); / 工 入 (include) adodb.inc.php 文 件 
$conn = ADONewConnection('access'); /连接 Access 数据 库 


$conn-> PConnect("Driver={Microsoft Access Driver (*.mdb)};Dbq=d:\mydb.mdb;Uid=Admin;Pwd=;"); 
?> 


其 中 ，di\mydb.mdb 表示 数据 库 所 在 的 文件 名 。 
19.4.2 ”操作 数据 库 函 数 


区 1 视频 讲解 :光盘 \TM\Video\ 第 19 章 \ 操 作 数 据 库 函数 .exe 
1. execute($sql[,$inputarr=false]) 
函数 作用 : 执行 SQL 语句 ， 并 返回 一 个 结果 集 (ADORecordSet 对 象 ) ;如 果 失 败 则 返回 false。 


人 注意 无 论 是 执行 SELECT 语句 ， 还 是 INSERT 或 UPDATE 语句 ， 只 要 执行 成 功 ，executeO 函 数 都 
会 返回 一 个 结果 集 。 


参数 说 明 如 下 。 

$sql: 要 执行 的 SQL 语句 。 

$inputarr: 用 来 作为 传 入 的 结合 变量 (Binding variables) 。 

没有 变量 $inputarr 时 ，$sql 为 普通 的 SQL 语句 。execute0 函 数 的 格式 为 : 

$connect->execute('select * from tb_object where id = 1); 

当 使 用 变量 $inputarr 时 ，execute0) 函 数 的 格式 为 : 

$connect -> execute('select * from tb_object where id = ?,array($vl)); 

结合 变量 可 以 加 速 SQL 指令 编译 及 读 取 的 速度 ， 产 生 较 佳 的 效能 。 

例 19.3 应 用 execute0 函 数 的 第 二 个 参数 $inputarr, 为 WHERE 子 句 补充 上 完整 的 条 件 , 查询 出 数据 表 
中 了 D 等 于 2 的 记录 。 (实例 位 置 : 光盘 \TM\Instance\19\19.3) 


代码 如 下 : 
<?php 
include_once (../adodb5/adodb.inc.php'); 1/ 载 入 (include) adodb.inc.php 文 件 
$ADODB_FETCH_MODE =ADODB_FETCH_ASSOC; // 设 置 字段 为 结果 集 索引 
$conn = ADONewConnection(mysql); /建立 连接 
$conn -> PConnect(localhost,root,"111',"db_database19'); /连接 数据 库 
$conn -> execute('set names gb23127); // 设 置 编码 格式 
$sqlstr = 'select * from tb_object where id = ?5 /创建 SQL 语句 
Srst = $conn -> execute($sqlstr,'2') or die('connect error); /执行 SQL 语句 ， 查 询 出 id 等 于 2 的 数据 
while(!$rst -> EOF){ /iwhile 语句 循环 输出 结果 
echo $rst -> fields['bigclass'"].'=>'.$rst -> fields[d_time']… // 输 出 查询 结果 
Srst -> movenext(); /指针 下 移 
} 
Srst -> close(); /关闭 连接 


S$conn -> close(); 
?> 


结果 为: 图 书 开发 =>2012-11-13 16:46:48。 
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2. cacheExecute([$sec,]$sql[,$inputarr=false]) 

函数 作用 : 除了 具有 execute0 函 数 的 功能 外 , 还 可 将 查询 结果 保存 到 缓存 中 。 如 果 以 后 还 有 相同 的 查询 ， 
即 可 直接 从 缓存 中 获取 。 

参数 说 明 如 下 。 

Ssec: 查询 结果 集 在 缓存 中 被 保存 的 时 间 。 

$sql: 执行 的 select 查询 语句 。 这 里 只 能 是 select 语句 。 

$inputarr: 结合 变量 。 

3. selectLimit($sql[,$numrows=-1][,$offset=-1][,$inputarr=false]) 

函数 作用 : 返回 一 个 指定 起 始 位 置 和 记录 数 的 结果 集 。MySQL 数据 库 支持 limit 查询 功能 ， 但 Access 
等 数据 库 系 统 并 不 支持 limit。ADODB 考虑 到 了 这 一 点 ， 模 拟 了 一 个 和 limit 功能 类 似 的 函数 。 这 里 参数 的 
位 置 和 MySQL 数据 库 中 limit 的 参数 位 置 不 太一 样 。 使 用 时 需要 注意 。 

参数 说 明 如 下 。 

$sql: 要 执行 的 select 查询 语句 。 

$numrows: 是 要 查询 的 记录 数 。 如 果 该 值 为 -1， 则 函数 会 一 直 查 询 到 最 后 一 条 记录 。 

$offset: 表示 从 第 几 条 记录 开始 计算 。 

$inputarr: 结合 变量 。 

例 19.4 使 用 selectlimit0 函 数 查询 数据 ， 从 第 $offset 条 记录 开始 查 起 ， 返 回 $Snumrows 条 记录 ， 最 后 输 
出 数据 。《〈 实 例 位 置 ， 光盘 \TMNInstance\l9\19.4) 


代码 如 下 : 
<?php 
include_once 'conn/conn.php'; // 载 入 数据 库 连 接 文件 
$sql = 'select * from tb_object'; // 要 查询 的 SQL 语句 
$numrows = 2; /查找 两 条 记录 数 
Soffset = 1; /从 第 一 条 记录 开始 计算 
Srst = $conn -> selectlimit($sql,$numrows,$offset) or die('execute error); /调用 selectlimit() 函 数 
while(l$rst -> EOF){ // 如 果 不 是 最 后 一 条 记录 , 循环 
echo $rst -> fields['id].=>'.$rst -> fields[bigclass']; /| 输出 id 和 字段 值 
echo '<br>"; 
S$rst -> movenext(); /指针 移动 一 位 
)} 
?> 


结果 为 : 网 络 =>2012-11-19 11:16:32 电器 =>2012-11-25 14:56:21。 

4. CacheSelectLimit([$sec,] $sql[, $numrows=-1][,S$offset=-1][,$inputarr=false]) 

函数 作用 : 除了 具有 selectlimit0 函 数 的 功能 外 ， 还 将 查询 结果 保存 到 缓存 当中 。 如 果 以 后 有 相同 的 查 
询 ， 将 直接 从 缓存 中 获取 。 

参数 含义 同上 。 

5. CachFlush() 

函数 作用 : 清除 所 有 ADODB 数据 库 的 缓存 。 

6. getUpdateSQL(&$rs, $arrFields, $forceUpdate=false) 

函数 作用 : 更 新 记录 信息 。 

参数 说 明 如 下 。 

Srs: 要 更 新 的 结果 集 。 


@ 


SarrFields: 更 新 字段 及 内 容 。 
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$forceUpdate: 如 果 S$forceUpdate 为 rue， 无 论 $SarrFields 和 Srs 的 值 是 否 相 等 都 将 更 新 。 
例 19.5 本 例 中 首先 查找 ID 等 于 2 的 记录 ， 然 后 创建 要 更 新 字段 的 数组 ， 最 后 将 该 数组 内 容 通 过 


getUpdateSQLO 函 数 更 新 到 数据 库 中 。 
代码 如 下 : 
<?php 
include_once ('../adodb5/adodb.inc.php’); 
S$conn = ADONewConnection( mysql); 


$conn -> PConnect('localhost','root','root','db_database19"); 


$conn -> execute('set names gb2312"); 
S$sqlstr = 'select * from tb_object where id = 2"; 


Srst = $conn -> execute($sqlstr) or die('Error: .$conn -> errorMsg()); 
echo ' 修 改 前 的 数据 ， 时 间 为 :".$rst -> fields['d_time'].'<br>"; 


Sfields = array(); 
Sfields['d_time'] = date("Y-m-d H:i:s"); 


(实例 位 置 ， 光盘 \TM\Instance\19\19.5) 


1/ 载 入 (include) adodb.inc.php 文 件 
// 建 立 连接 
// 连 接 数据 库 
// 设 置 编码 格式 
/声明 SQL 语句 
/执行 SQL 语句 
/输出 数据 库 中 的 时 间 
/声明 一 个 空 数组 $fields 
// 将 当前 时 间 存 到 数组 中 


Supdate = $conn -> getUpdateSQL($rst,$fields) or die(update error: '.$conn -> errorMsg());// 更 新 数据 
/| 执行 更 新 


$conn -> execute($update); 
echo "修改 后 的 数据 ， 时 间 为 : 
Srst = $conn -> a 
echo $rst -> fields['d_time']; 
Srst -> close(); 
S$conn -> close(); 

?> 


结果 为 : 更 新 前 的 时 间 : 2012-11-19 11:16:32 
7. getlnsertSQL(&$rs, $arrFields) 
函数 作用 : 添加 新 记录 。 
参数 说 明 如 下 。 
$rs: 要 添加 记录 的 结果 集 。 
SarrFields: 新 记录 的 字段 值 及 对 应 字段 名 。 
例 19.6 使 用 getInsertSQLO 函 数 向 表 tb_object 中 插入 
实例 代码 如 下 : 
<?php 
include_once ('../adodb5/adodb.inc.php'); 
$ADODB_FETCH_MODE = ADODB_FETCH_ASSOC.; 


更 新 后 的 时 间 : 


/执行 查询 
// 输 出 更 新 后 的 信息 
/关闭 连接 


2012-11-24 07:31:44。 


-条 新 数据 .( 实 例 位 置 :光盘 \TMN\Instance\19\19.6) 


/1 载 入 (include) adodb.inc.php 文 件 
/设置 结果 集 按照 字段 名 称 为 索引 进行 存 取 


$conn = ADONewConnection('mysql); /建立 连接 

$conn -> PConnect(localhost,root,"111',db_database19'); /连接 数据 库 

$conn -> execute('set names gb2312'); /设置 编码 格式 

$sqlstr = 'select * from tb_object where id = 0'; /| 生成 SQL 语句 

Srst = $conn -> execute($sqlstr) or die('Error: '.$conn -> errorMsg()) ; /执行 SQL 语句 
Sfields = array(); /创建 一 个 数组 
Sfields['bigclass'] = ' 办 公用 品 ' /向 数组 中 添加 数据 
$fields[d_time] = date("Y-m-d H:i:s"); 

Sinsert = $conn -> getlnsertSQL($rst,$fields) or die(update error: '.$conn -> errorMsg()); /添加 新 数据 
S$conn -> execute(S$insert); /执行 添加 
echo ' 添 加 的 数据 为 : '; 

$showsql = "select * from tb_object where d_time = ".$fields[d_time].""; /创建 SQL 语句 

Srst = $conn -> execute($showsql); /查找 刚 创建 的 记录 
foreach($rst -> fields as $nm => $vl) // 使 用 foreach 输出 新 记录 


echo $nm.'=>".$vl.' "; 
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Srst -> close(); /关闭 连接 
S$conn -> close(); 
?> 


结果 为 :添加 的 数据 为 ，id=>78 bigclass=> 办 公用 品 d_time=>2012-11-24 07:38:50。 

8. DBDate($date) 

函数 作用 : 转换 不 同 数据 库 之 间 的 日 期 格式 。 例 如 ，MySQL 使 用 的 时 间 格 式 为 Y-m-d，Oracle 则 为 
D-M-Y。 

参数 $date 是 要 存储 的 时 间 。 

9. qstr($string) 

函数 作用 : 使 用 引号 来 处 理 字符 串 。 例 如 ， 一 个 字符 串 包 含 了 单 引号 (') ， 在 MySQL 中 可 以 直接 
使 用 “'” 表 示 , 但 在 其 他 的 数据 库 中 ， 则 使 用 两 个 单 引 号 (") 表示 。 使 用 qstr0 函 数 可 以 处 理 和 解决 这 个 
问题 。 

参数 $string 为 要 被 处 理 的 字符 串 。 

例 19.7 使 用 DBDate0 和 qstr0 函 数 向 表 tb_object 中 插入 一 条 数据 。〈 实 例 位 置 ， 光 盘 \TMNInstance\ 
19\19.7) 


代码 如 下 : 

<?php 
include_once 'conn/conn.php'; // 载 入 数据 库 连接 文件 
$bigclass = $conn -> qstr(" 建 筑 耗 材 "); /用 qstr() 函 数 处 理 带 引号 的 字 串 
$d_time = $conn -> DBDate(date(Y-m-d H:is'")); /用 DBDate() 函 数 处 理 日 期 时 间 
$sql = 'insert into tb_object(bigclass,d_time) value('.$bigclass.,.$d_time.) /生成 insert 语句 
S$rst = $conn -> execute($sql) or die($conn -> errorMsg()); /执行 insert 语句 
if($rst) 

echo ' 添 加 成 功 "; /添加 成 功 
Se 


结果 为 :添加 成 功 。 


4 
-说 日 
a 明 在 本 实例 中 ,为 了 使 代码 整洁 ,便于 对 使 用 数据 库 的 管理 ,将 连接 数据 库 的 操作 定义 到 conn.php 
文件 中 ， 在 使 用 时 只 要 通过 include_once 语句 进行 调用 即 可 。 


10. Affected_rows() 
函数 作用 : 返回 最 后 更 新 或 被 删除 的 记录 数 。 如 果 数 据 库 不 支持 ， 则 返回 false。 


例 19.8 对 id 为 2 的 记录 进行 修改 ,将 该 记录 的 d_time 字 段 值 设 为 当前 时 间 , 然后 调用 Affected_rows( 
函数 查看 返回 结果 。 (实例 位 置 : 光盘 \TM\Instance\19\19.8) 


代码 如 下 : 
<?php 
include_once 'conn/conn.php'; // 载 入 数据 库 连接 文件 
$sqlstr = 'update tb_object set d_time = now() where id = 2'; JSQL 更 新 语句 
Srst = $conn -> execute($sqlstr) or die(update error: '.$conn -> errorMsg()); 。 // 执 行 更 新 语句 
echo 更 新 记录 数 :'.$conn -> Affected_rows(); /查看 更 新 的 记录 数 
> 
结果 为 : 更 新 记录 数 : 1。 
11. Insert_ID() 


函数 作用 : 返回 最 后 插入 的 记录 ID 值 。 如 果 数 据 库 不 支持 该 函数 ， 则 返回 false。 


@ 
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19.4.3 ”控制 结果 集 存 取 方式 


铬 1 视频 讲解 : 光盘 \TM\Video\ 第 19 章 \ 控 制 结果 集 存 取 方式 .exe 
1. $ADODB_COUNTRECS 
当 该 变量 为 tue 时 , RecordCount0 函 数 会 在 数据 库 驱 动 不 支持 select 指令 返回 的 记录 总 数 时 进行 自动 模 
拟 , 返回 记录 数 ， 默 认 值 为 rue。 每 次 进行 查询 操作 时 都 会 自动 检查 该 变量 的 值 。 由 于 这 个 功能 十 分 消耗 内 
存 ， 所 以 不 建议 使 用 。 
2. $ADODB_CACHE_DIR 
该 变量 用 于 设置 缓存 目录 。 一 般 在 使 用 缓存 函数 时 会 用 到 该 变量 ， 如 CacheExecute0 〇 函数 。 
3. $ADODB_FETCH_MODE 
该 变量 决定 结果 集 以 哪 种 方式 进行 存 取 。 其 有 4 个 值 : 
(1) define( ADODB FETCH DEFAULT'0) 
(2) define( ADODB FETCH NUM,'1) 
结果 集 按照 字段 序号 为 索引 进行 存 取 ， 如 Srst -> fields[0]、Srst -> fields[1]。 
(3) define( ADODB FETCH ASSOC'2) 
结果 集 按照 字段 名 称 为 索引 进行 存 取 ， 如 S$rst -> fields[bigclass]、Srst -> fields[id]。 
(4) define( ADODB FETCH BOTH',3) 
结果 集 同 时 支持 (2) 、 (3) 两 种 方式 。 如 果 未 设置 该 变量 ， 默 认 值 为 (1) ， 不 同类 型 的 数据 库 驱 动 
默认 值 也 不 相同 。 这 里 不 建议 使 用 变量 (4) ， 因 为 有 很 多 驱动 并 不 支持 。 
例 19.9 仍 使 用 例 19.1 中 的 内 容 ， 只 是 增加 了 一 个 公共 变量 ， 将 结果 集 以 字段 名 作为 索引 进行 输出 。 
(实例 位 置 ， 光 盘 \TM\Instance\19\19.9) 


代码 如 下 : 
<?php 
include_once ('../adodb5/adodb.inc.php'); // 载 入 (include) adodb.inc.php 文 件 
$ADODB_FETCH_MODE = ADODB_FETCH_ASSOC.; // 设 置 公共 变量 值 
S$conn = ADONewConnection(mysql); /建立 连接 MySQL 数据 库 
Connect($host,[$user],[$password],[$database]) // 连 接 数据 库 
S$conn -> PConnect(localhost,root,"111',db_database19'); /| 连接 指定 的 数据 库 
$conn -> execute('set names gb2312'); /| 设置 编码 格式 
S$rst = $conn -> Execute('select * from tb_object') or die(' 执 行 错误 ');// 执 行 查询 语句 
while(!$rst -> EOFX{ /while 语句 循环 输出 结果 
echo $rst -> fields['id"].'=>'".$rst -> fields[bigclass].' '; // 循 环 输出 记录 
Srst -> movenext(); /指针 下 移 
上 
S$rst -> close(); /关闭 连接 方法 
$conn -> close(); /关闭 数据 库 


?> 


结果 为 : 1=> 图 书 开发 2=> 网 络 3=> 电 器 78=> 办 公用 品 79=> 网 络 。 
19.4.4 ”操作 结果 集 函 数 


如 4 视频 讲解 : 光盘 TM\Video\ 第 19 章 \ 操 作 结 果 集 函数 .exe 
当 使 用 execute0 函 数 执行 SQL 指令 时 ， 会 回 传 一 个 结果 集 。 该 结果 集 实 际 上 是 一 个 ADORecordSet 对 
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象 。 通 过 对 该 对 象 的 控制 ， 可 以 对 结果 进行 各 项 操作 。 

1. GetArray([$number_of_rows]) 

函数 作用 : 返回 从 当前 指针 指向 的 记录 开始 ， 到 ($number_of rows-1) 行 的 全 部 记录 的 数组 。 

参数 $number_of rows 表示 指定 的 记录 行 。 如 果 没 有 给 出 ， 则 一 直到 EOF 才 停 止 。 

2. UserDate(S$str, [Sfmt]) 

函数 作用 : 将 日 期 字符 串 $str 转换 为 变量 $fmt 设置 的 日 期 格式 。 

参数 说 明 如 下 。 

回 $str: 日 期 格式 的 字符 串 。 

$ftmt:， 要 转换 的 日 期 格式 ， 默 认 值 为 Y-m-d。 

3. UserTimeStamp($str, [$fimil) 

函数 作用 : 将 时 间 字 符 串 $str 转换 为 变量 $fmt 设置 的 时 间 格 式 。 

参数 说 明 如 下 。 

$str: 时 间 字 符 串 。 

$fnt:， 时 间 格式 设置 字 串 ， 默 认 值 为 Y-m-d His。 

4. MoveNext() 

函数 作用 : 将 ADORecordSet (结果 集 ) 的 指针 下 移 一 位 。 如 果 成 功 ， 则 返回 tue， 和 否则 返回 false。 

5. Move($to) 

函数 作用 : 将 ADORecordSet (结果 集 ) 的 指针 移动 到 指定 位 置 。 如 果 $to 等 于 0， 则 指针 指向 结果 集 的 
第 一 条 数据 ， 如 果 $to 的 值 大 于 结果 集 ， 则 指针 指向 最 后 一 条 数据 。 注 意 ， 这 里 的 变量 $to 只 能 是 绝对 定位 。 

6. MoveFirst() 

函数 作用 : 指针 移动 到 第 一 条 数据 ， 等 同 于 Move(0)。 

7. MoveLast() 

函数 作用 : 指针 移动 到 最 后 一 条 数据 ， 等 同 于 Move(RecordCountO-1)。 

8. FetchRow() 

函数 作用 : 返回 当前 指针 指向 的 记录 的 数组 ,如果 是 EOF， 则 回 传 false。FetchRow0 不 要 和 MoveNext() 
混用 。 

9. FetchField($column_number) 

函数 作用 : 返回 一 个 对 象 ， 包 含 字段 $column_number 的 名 称 、 空 间 长 度 等 相关 信息 。 

参数 $column_number 表示 要 查看 的 字段 名 称 。 

10. FetchNextObject([$toupper=true]) 

函数 作用 : 返回 当前 指针 所 指向 的 记录 的 对 象形 式 ， 并 且 指 针 自 动 下 移 一 行 。 

参数 gtoupper: 如 果 等 于 trtue， 则 字段 名 为 大 写 形式 。 

11. FieldCount() 

函数 作用 : 返回 结果 集中 的 字段 数 。 

12. RecordCount() 

函数 作用 : 返回 结果 集中 的 记录 数 。 
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13. CurrentRow() 

函数 作用 : 返回 当前 指针 所 指 的 记录 序号 。 第 一 条 记录 用 0 表示 。 

14. MetaType($nativeDBType [,$field_max_length],[$fieldobj]) 

函数 作用 : 标准 化 不 同 数据 库 系 统 下 的 数据 类 型 的 表示 法 。 处 理 不 同 的 数据 库 系 统 有 一 个 问题 : 每 一 
个 数据 库 系 统 对 于 相同 的 数据 类 型 会 有 不 同 的 表示 法 ， 如 支持 超 长 文本 的 类 型 ， 有 的 表示 为 text， 有 的 表示 
为 long character， 还 有 clob 等 。MetaType0 函 数 可 以 将 这 几 种 表示 法 统一 起 来 ， 如 用 大 写字 母 D 来 表示 所 
有 的 日 期 类 型 ， 用 大 写字 母 T 表 示 所 有 的 时 间 类 型 。 

参数 说 明 如 下 。 

加 ”SnativeDBType: 数据 表 中 的 数据 类 型 。 

回 $fields max _length: 字段 的 最 大 长 度 。 

加 ”$fieldobj: 字段 对 象 。 

MetaType0 函 数 支持 的 标准 化 数据 类 型 如 下 。 


C: char、varchar 等 字符 类 型 。 

X: text、long character 等 长 类 型 。 

B: blob 或 binary image。 

D: 日 期 date。 

T: 时 间 timestamp。 

L: 迪 辑 类 型 和 位 类 型 。 

N: 包含 编号 、 整 型 、 浮 点 型 等 数字 类 型 。 
R: 包含 序列 、 自 动 增加 整数 等 序列 类 型 。 


例 19.10 ”通过 FieldCount0 函 数 获取 结果 集 记录 数 ,然后 通过 for 循环 输出 数据 ， 并 对 每 个 字段 类 型 进 
行 判断 , 如 果 类 型 为 T, 则 使 用 DBDate0 函 数 格式 化 输出 时 间 , 其 他 类 型 则 直接 输出 。( 实 例 位 置 : 光盘 \IM\ 
Instance\19\19.10) 


代码 如 下 : 
<?php 
include_once 'conn/conn.php'; // 载 入 数据 库 连 接 文件 
$sqlstr = 'select * from tb_object where id = 1'; //SQL 查询 语句 
Srst = $conn -> execute($sqlstr) or die('error: '.$conn -> errorMsg()); 。 // 执 行 查询 语句 
if(false != $rst)f 1/ 如 果 有 查询 结果 
for($i = 0; $i < $rst -> FieldCount(); $i++X{ /| 循环 输出 各 个 字段 
Sfields = $rst -> FetchField($i); /生成 字段 信息 对 象 
$type = $fields -> type; // 从 对 象 中 获取 字段 的 类 型 信息 
echo '=>" 
echo $rst -> metaType($type,-1, $fields); /输出 字段 标准 类 型 
if($rst -> metaType($type,-1,$fields) == "T") 1/ 如果 标准 类 型 为 
echo '('.$conn -> DBDate($rst -> fields[$i]).")'; // 使 用 DBDate 函数 格式 化 时 间 
else 
echo '(.$rst -> fields[$i].")'; // 如 果 是 其 他 类 型 ， 直 接 输 出 
} 
} 
?> 


结果 为 : =>R(1)=>C( 图 书 开发 )=>T(2012-11-13 16:46:48')。 
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19.4.5 ”处理 事务 函数 


1. BeginTrans() 

函数 作用 : 开启 事务 处 理 。 该 函数 和 下 面 介绍 的 两 个 函数 都 应 用 在 PHP 的 事务 处 理 中 。 

2. CommitTrans() 

函数 作用 : 如 果 成 功 完成 数据 库 操作 ， 则 执行 CommitTrans0 〇 函数。 

3. RollbackTrans() 

函数 作用 :结束 一 次 操作 ,恢复 操作 前 的 所 有 改变 。 如 果 成 功 则 返回 true, 如 果 服 务 器 不 支持 则 返回 false。 


19.4.6 生成 HTML 表格 函数 


rs2html($rst,[$table_attributes], [$col_titles]) 

函数 作用 : 返回 一 个 HTML 表格 格式 的 结果 集 。 要 使 用 该 函数 ， 需 要 载 入 tohtmlinc.php 文件 。 

参数 说 明 如 下 。 

S$rst: 要 返回 的 结果 集 。 

S$table_attributes: 对 表格 参数 及 属性 的 设置 。 

$col_ titles: 对 字段 的 重新 命名 。 

例 19.11 应 用 rs2html0 函 数 直接 输出 返回 的 结果 集 $rst， 并 设置 表格 的 属性 。《〈 实 例 位 置 ， 光盘 \TM 
Instance\19\19.11) 


代码 如 下 : 
<?php 
include_once 'conn/conn.php'; // 载 入 数据 库 连 接 文件 
include_once '../adodb5/tohtml.inc.php'; / 载 入 tohtml.inc.php 文 件 
$rst = $conn -> execute('select * from tb_object'); /| 返回 查询 结果 集 
rs2html(S$rst,' width="350" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" bgcolor="#FFO000", 
array(' 序 号 ,类 别 ', 添 加 时 间 ')); // 输 出 结果 集 
?> 
运行 结果 如 图 19.1 所 示 。 en po | 
19.4.7 ”生成 下 拉 列 表 框 函 数 ene Be | 


区 4 视频 讲解 : 光盘 \TM\Video\ 第 19 章 \ 生 成 下 拉 列 
表 框 函数 .exe 


1. getMenu($name, [$default_str = "], [$blank = true]， 图 19.1 应 用 rs2html0 函 数 直接 输出 返回 的 结果 集 
[$multiple_select = false], [$size = 0], [$moreAttr = "]) 

函数 作用 : 返回 一 个 HTML 下 拉 列 表 框 (select) 。 结 果 集 的 第 一 列 fields[0]) 被 显示 到 下 拉 列 表 中 ， 
第 二 列 (fields[1]) 被 设置 为 <option> 标 签 的 value 属性 。 

参数 说 明 如 下 。 

加 ”$name: 下 拉 列 表 框 的 名 称 。 

Sdefault_str: 如 果 $default_str 的 值 等 于 其 中 一 个 fields[0]， 那 么 这 个 fields[0] 就 是 该 下 拉 列 表 的 默 
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认 选 项 。 这 相当 于 为 fileds[0] 添 加 了 属性 <selected>。 
S$blank: 该 参数 的 默认 值 为 tue。 在 生成 下 拉 列 表 框 时 , 第 一 行 选项 默认 为 空 , 即 <option></option>。 
如 果 为 false， 则 没有 默认 的 空 选项 。 
$multiple_select: 是 否 支 持 多 选 。 默 认 值 为 false， 即 不 允许 多 选 。 
$size: 列表 框 选取 的 大 小 。 默 认为 0， 生 成 一 个 下 拉 列 表 ; 如 果 $size 的 值 大 于 0， 则 生成 一 个 列表 框 。 
$moreAttr: 该 参数 可 以 添加 列表 框 的 属性 ， 如 添加 一 个 JavaScript 脚本 等 。 
. getMenu2($name, [$default_str="], [$blank1stltem=true], [$multiple_select=false], [$size=0], 
[$moreAttr="]) 
函数 作用 和 参数 含义 同上 。 该 函数 和 GetMenu0) 函 数 的 不 同 之 处 是 ，getMenu0 函 数 中 的 参数 $default_str 
和 fields[0] 比 较 ， 如 果 相 等 ,那么 fields[0] 就 是 下 拉 列 表 的 默认 选项 ; 而 getMenu20 函 数 中 的 参数 $default_str 
则 是 和 fields[1] 进 行 比较 ， 其 他 不 变 。 
例 19.12 本 例 中 首先 使 用 查询 语句 和 execute0 函 数 返 回 一 个 结果 集 ， 然 后 在 一 个 form 表单 中 使 用 
getMenu20 函 数 将 结果 集 输出 到 下 拉 列 表 框 中 。 当 单 击 “ 提 交 ” 按 钮 时 ， 通 过 $_ POST[] 预 定义 变量 获取 下 拉 
列表 框 提交 的 值 。〈 实 例 位置 ; 光盘 \TM\Instance\19\19.12》 
代码 如 下 : 
<?php 
include_once 'conn/conn.php'; // 载 入 数据 库 连 接 文件 
$sqlstr = 'select bigclass,id from tb_object'; JSQL 查询 语句 
Srst = $conn -> execute($sqlstr) or die('execute error: '.$conn -> errorMsg()); 。” // 执 行 查询 语句 ， 返 回 结 果 集 
?> 
<!- 提交 表单 --> 
<form method='post action="> 
请 选择 购买 商品 的 类 别 : 

<!-- 调用 getMenu2() 函 数 ”--> 
<?php echo $rst -> getMenu2('bigclass',$_POST[bigclass],false); ?> 
<input type='submit name='submit value=' 提 交 ' /> 


[sl 


图 回回 


De 


</form> 

<I-- > 

<?php 

if(isset($_POST[bigclass])X{ // 判 断 表单 是 否 被 提交 
echo ' 您 选择 的 商品 类 别 是 :'.$_POST[bigclass]; // 输 出 提交 的 表单 值 

} 

?> 


运行 结果 如 图 19.2 所 示 。 


19.4.8 ”实现 分 页 功能 函数 


名 ml 视频 讲解 :光盘 \TM\Video\ 第 19 章 \ 实 现 分 页 功能 函 ER 


es 图 19.2 使 用 getMenu20 函 数 输 出 下 拉 列 表 杠 
1. PageExecute($sql, $nrows, $page) 
函数 作用 : 分 页 功能 函数 。 
参数 说 明 如 下 。 
$sql: SQL 查询 语句 。 
Snrows: 每 页 显示 的 记录 数 。 
Spage: 保存 当前 页 数 ， 默 认为 1。 
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2. CachePageExecute($sec, $sql, $nrows, $page) 

函数 作用 : 分 页 显示 函数 ， 同 时 将 显示 后 的 记录 放 到 缓存 中 。 在 $sec 秒 内 ， 如 果 是 重复 查看 ， 将 从 组 
存 中 读 取 数据 。 

参数 $sec 表示 数据 在 缓存 中 保留 的 时 间 。 其 他 参数 见 PageExecute0 函 数 。 

3. AbsolutePage($page=-1) 

函数 作用 : 返回 当前 的 页 数 。 要 和 PageExecute() 函 数 配 合 使 用 ， 用 于 分 页 。 

4. AtFirstPage($status='") 

函数 作用 : 如 果 当前 是 第 一 页 ， 返 回 tue。 要 和 PageExecute0 函 数 配 合 使 用 ， 用 于 分 页 。 

5. AtLastPage($status=") 

函数 作用 : 如 果 当 前 是 最 后 一 页 ， 返 回 tue。 要 和 PageExecute() 函 数 配 合 使 用 ， 用 于 分 页 。 

6. ADODB_Pager($conn, $sql, $id = 'adodb', $showPageLinks = false) 


函数 说 明 : 该 函数 是 adodb_pager 类 的 构造 函数 ， 通 过 类 中 的 render0 函 数 可 以 实现 一 个 小 巧 的 分 页 操 
作 。 如 果 要 使 用 这 个 功能 ， 需 要 载 入 adodb_paper.inc.php 页 。 
参数 说 明 如 下 。 


$conn: 数据 库 连 接 对 象 。 

$sql: 执行 的 SQL 语句 。 

$id: 每 个 分 页 的 id 号 。 

$showPageLinks: 是 否 显示 各 个 页 的 链接 ， 默 认为 false。 

例 19.13 使 用 adodb pager 类 将 表 tb_object 中 的 记录 分 页 显示 出 来 。 在 本 实例 中 对 载 入 文件 


adodb_pager.inc.php 进行 了 修改 。〔 实 例 位 置 ， 光盘 \TM\Instance\19\19.13) 
修改 的 代码 如 下 : 
var $first = '<code> 首 页 </code>"; 
var $prev = '<code> 上 一 页 </code>'; 
var $next = '<code> 下 一 页 </code>'; 
var $last = '<code> 尾 页 </code>'; 
上 述 修改 是 为 了 将 翻 页 改 成 中 文 连接 。 同样 , 在 应 用 的 过 程 中 还 可 以 修改 adodb_pager.inc.php 文件 中 的 
其 他 内 容 ， 如 表格 的 背景 、 宽 度 ， 以 及 边框 的 宽度 、 颜 色 等 。 
本 实例 的 代码 如 下 : 


<?php 
include 'conn/conn.php'; // 载 入 数据 库 连 接 文件 
include_once '../adodb5/adodb_pager.inc.php'; // 载 入 adodb_pager.inc.php 文 件 
/声明 对 象 */ 


$pager = new ADODB_Pager($conn,"select id,bigclass as ' 大 类 ', d_time as ' 添 加 时 间 ' from tb_object"); 
/* 调用 $Render 函数 确定 每 页 显示 多 少 */ 
$pager -> Render(3); 

?> 


运行 结果 如 图 19.3 所 示 。 
19.4.9 ”错误 处 理 函 数 


ADODB 的 优势 是 可 以 让 各 种 数据 库 函 数 统一 起 来 ， 对 相同 的 数据 进行 相同 的 操作 。 但 在 实际 操作 时 ， 
难免 会 出 现 各 种 问题 。 下 面 介绍 几 种 错误 处 理 及 调试 的 方法 。 
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1. debug 

如 果 变 量 被 设 定 为 tue， 当 有 输出 操作 时 ， 同 时 也 输出 调试 信息 。 

2. errorMsg() 

函数 作用 : 返回 最 后 的 状态 或 出 错 信息 。 即 使 没有 错误 发 生 ， 也 会 返回 一 个 字 串 。 所 以 ， 一 般 情 况 下， 
只 有 在 发 生 错误 时 〈 返 回 false) 才 调用 该 函数 。 在 启用 debug 变量 后 ， 只 要 有 错误 发 生 就 会 自动 调用 该 函数 。 

例 19.14 人 为 地 造成 一 个 错误 ， 将 SQL 语句 中 的 from 写成 form， 来 看 一 下 输出 的 错误 信息 。( 实 例 
位 置 ， 光盘 \TM\Instance\19\19.14) 


<?php 
include_once 'conn/conn.php'; // 载 入 数据 库 连接 文件 
S$conn -> debug = true; /开启 调试 
$sql = 'select * form tb_object: /SQL 查询 语句 
Srst = $conn -> execute($sql) or die($conn -> errorMsg()); // 调 用 查询 语句 
和 


运行 结果 如 图 19.4 所 示 。 


debug 弹出 的 错误 信息 元 


En 
EF 


19.3 ”adodb_pager 类 用 于 显示 记录 分 页 图 19.4 ADODB 调试 
可 以 发 现 ，debug 弹出 的 错误 提示 中 ， 已 经 包含 了 errorMsg0 函 数 的 错误 信息 。 


19.5 实 战 


铬 1 视频 讲解 :光盘 \TM\Video\ 第 19 章 \ 实 战 .exe 


19.5.1 实现 分 页 


例 19.15 除了 可 以 使 用 adodb_pager.inc.php 中 的 类 实现 分 页 效果 外 , 用 ADODB 类 库 中 默认 的 几 个 函 
数 同样 可 以 实现 分 页 效果 。 分 页 主要 使 用 PageExecute0 、AbsolutePage0 、AtFirstPage0 、AtLastPage0 和 
LastPageNo0 5 个 函数 来 完成 。〈 实 例 位置 ， 光盘 \TM\Instance\19\19.15) 


实例 代码 如 下 : 

<table width="384" border="0" cellpadding="0" cellspacing="0"> 
<tr><td height="30"> 

<?php 
include_once 'conn/conn.php'; /三 入 数据 库 连接 文件 
include(./adodb5/tohtmlinc phpy): / 厂 入 tohtml.inc php 文件 
$sql = 'select * from tb_object'; /查询 语句 
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Snum = 2; 
if(isset($_GETI'n_page'])X{ 
$c_page =$_GET[n_page]; 
jelsef 
$c_page =1; 


} 
Srst = $conn -> PageExecute($sql,$num,$c_page); 
if(false != $rst){ 
if(!$rst -> AtfirstPage()X{ 
?> <!-- 输出 向 上 翻 页 超 链 接 --> 
<a href ="<?php echo '?n_pge=1' ?>"> 首页 </a> 


/| 每 页 显示 的 记录 数 

/判断 当前 页 码 

// 将 $n_page 赋 给 变量 $c_page 
// 初 始 化 变量 $c_page 

/| 执行 PageExecute() 函 数 


// 如 果 当前 页 不 是 首页 


<a href ="<?php echo '?n_page=".($rst -> AbsolutePage() - 1); ?>"> 上 一 页 </a> 


<|-- 一 > 
<?php 
} 
if(!$rst -> AtlastPage()}{ 1/ 如果 当前 页 不 是 尾 页 
?> 
<!-- 输出 向 下 翻 页 超 链 接 -> 
<a href = "<?php echo '?n_page='.($rst -> AbsolutePage() + 1); ?>"> 下 一 页 </a> 
<a href ="<?php echo '?n_page='.($rst -> LastPageNo());?>"> 尾 页 </a> 
<L-- ”一 一 一 一 一 一 -一 “一 > 
<?php 
} 
i 
</td></tr> 
<tr><td> 
<?php 
rs2html($rst,width=350 border="1" bgcolor="#FF0000",array('ID",' 类 型 ',' 添 加 时 间 ")); 
?> 


<?php } ?> 
<tr><td height="30"> 


当前 是 第 <?php echo $rst -> AbsolutePage(); ?> 页 /一 共 是 <?php echo $rst -> LastPageNo(); ?> 页 


</td></tr> 
</table> 


运行 结果 如 图 19.5 所 示 。 
19.5.2 “处理 事务 


在 对 数据 库 进 行 操作 时 ， 由 于 发 生意 外 ， 导 致 数据 更 新 
到 一 半 时 就 停止 了 。 这 时 可 以 通过 事务 将 数据 恢复 到 更 新 以 
前 状态 。 下 面 就 来 看 一 个 通过 事务 实现 回 滚 的 实例 。 
例 19.16 在 ADODB 类 库 中 ， 通 过 BeginTransO 、 
CommitTrans0 和 RollbackTrans0 3 个 函数 实现 处 理事 务 。 
实例 代码 如 下 : 
<?php 
include_once 'conn/conn.php'; 
S$conn -> BeginTrans(); 
$sql = 'delete from tb_object where id = 2"; 
Srst = $conn -> execute($sql) or die(execute error: 
Snum = $conn -> Affected_rows(); 
if(false !== $rstX{ 
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"$conn -> ErrorMsg()); 
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图 19.5 分 页 技术 


(实例 位 置 ， 光盘 \TM\Instance\19\19.16) 


// 载 入 数据 库 连接 文件 

// 开 始 事务 处 理 

/SQL 删除 语句 

/| 执行 删除 语句 
/查看 被 更 新 的 记录 数 

/如果 $rst 不 为 假 


?> 


if($num != 0X{ 
S$conn -> CommitTrans(); 
echo ' 删 除 成 功 ! 
exit(); 
Jelse{ 
echo ' 没 有 数据 ， 或 数据 已 删除 '; 
exit(); 


} 

}else{ 
S$conn -> RollbackTrans(); 
echo ' 出 现 意外 。'; 

} 
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// 如 果 $num 不 为 0， 说 明 删 除 成 功 
/执行 提交 


/如 果 $num 为 0， 说 明 没有 删除 记录 


/如果 发 生意 外 
/| 执行 回 滚 操作 


本 实例 如 果 存 在 指定 的 数据 ， 则 执行 删除 后 返回 “删除 成 功 ! ”; 如 果 不 存 在 ， 则 输出 “没有 数据 ， 
或 数据 已 经 删除 ”; 如果 出 现 意外 ， 则 执行 回 滚 并 输出 


19.5.3 缓存 函 


现在 
器 负载 ， 


而 且 可 以 提高 被 搜索 引擎 搜索 到 的 几率 。 


“出 现 意 外 。” 


数 +ADODB 动态 生 成 静态 页 人 八 


E 很 多 站 点 都 采用 动态 生成 静态 页 面 的 方式 ， 这 样 不 仅 可 以 有 效 地 提高 站 点 的 访问 效率 ， 降 低 服务 


例 19.17 使 用 缓存 函数 和 ADODB 类 库 制作 能 够 动态 生成 静态 页 面 的 学 生成 绩 管理 系统 。 (实例 位 置 : 
光盘 \TM\Instance\19\19.17) 
实现 步骤 如 下 。 


(1) 本 实例 使 用 ADODB 技术 管理 MySQL 数据 库 ， 


为 了 提高 代码 重用 率 ， 体 现 PHP 面向 对 象 编程 的 


特点 ， 开 发 实例 时 创建 数据 库 连 接 类 建立 与 MySQL 数据 库 的 连接 。 
代码 如 下 : 
<?php 
class ConnDB 


var $dbType; 
var $host; 

Var $userName; 
var $password; 
var $dbName; 
var $isDebug; 
var $connlD; 


/| 数据 库 类 型 

/MySQL 数据 库 服务 器 地 址 
/用 户 名 

/密码 

/数据 库 名 

// 是 否 调试 

// 保 存 数据 库 连 接 标 识 


function ConnDB ($dbType = 'mysql', $host, $userName, $password, $dbName, $isDebug = false) 


‘ 
/构造 函数 ， 用 于 对 类 中 的 属性 进行 初始 化 
Sthis->dbType = $dbType; 
Sthis->host = $host; 
$this->userName = $userName; 
S$this->password = $password; 
S$this->dbName = $dbName; 
Sthis->isDebug = $isDebug; 


function getConnlD () 


require_once "library/adodb/adodb.inc.php'; 


ll 导入 ADODB 类 库 
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Sthis->connID = NewADOConnection($this->dbType): /建立 连接 
if ($this->dbType == 'mysql' || $this->dbType == 'mssql){ /MySQL 或 SQL Server 数据 库 
$this->connID->Connect($this->host, $this->userName, $this->password, $this->dbName); 
if ($this->dbType == 'mysql){ 
$this->connID->Execute('set names gbk'): /1 如果 是 MySQL 数据 库 则 设置 字符 集 


} 
} elseif ($this->dbType == 'access') { lAccess 数据 库 
Sthis->connID->Connect('Driver = {Microsoft Access Driver (*.mdb)}; Dbq =' . realpath($this-> 
dbName) . ';Uid = ' . $this->userName . ;Pwd = ' . $this->password. ";'"); 


}else{ 
return false; 
} 
Sthis->connID->debug = $this->isDebug; /| 是否 调试 
return $this->connID; 
} 
function closeConnlD () /关闭 与 数据 库 的 连接 


@sSthis->connID->Disconnect(); 
此 


站 

上 述 代 码 主 要 应 用 ConnDB 类 通过 ADODB 类 库 建立 与 MySQL 数据 库 的 连接 。 分 析 代 码 可 知 , ConnDB 
类 不 仅 可 以 建立 与 MySQL 数据 库 的 连接 ， 而 且 可 以 建立 与 Access 和 SQL Server 数据 库 的 连接 。ConnDB 
类 中 首先 使 用 构造 方法 对 类 中 的 属性 进行 实例 化 ， 然 后 创建 getConnID( 方 法 用 于 返回 数据 库 连 接 的 ID， 最 
后 创建 closeConnID() 方 法 释放 数据 库 连 接 资源 。 

(2) 建立 基于 ADODB 类 库 的 分 页 类 ， 用 于 实现 学 生成 绩 信 息 的 分 页 显示 。 

代码 如 下 : 

<?php 

class PageDB 


function pageData ($sql, $connlD, $pageSize, $page) 
4 


$arrayPagelnfo = array(); /定义 要 返回 的 数组 
if (isset($page) && $page {= "){ /| 判断 是 否 有 页 码 传 递 
$nowPage = $_GET[page]]; // 有 则 获取 页 码 的 值 
}else{ 
SnowPage = 1; // 没 有 则 默认 显示 第 一 页 
S$rs = $connlD->PageExecute($sql, $pageSize, $nowPage); 
$arrayData = $rs->GetRows(); /执行 查询 
if (count($arrayData) == 0 || $rs == false) { 
return false; // 如 果 出 错 或 没有 查询 到 则 返回 false 
}else{ 
SarrayPagelnfo[ data'] = $arrayData; /查询 到 的 数据 


SrsAll = $connID->Execute($sql); 
$countRs = count($rsAll->GetRows()); /总 记录 数 
$countPage = ceil($countRs / $pageSize); /总 页 数 
$arrayPagelnfo[ pageSize'] = $pageSize;  // 将 分 页 信息 保存 到 要 返回 的 数组 $arrayPagelnfo 中 
$arrayPagelnfo[ countRs'] = $countRs; 
$arrayPagelnfo[ page] = $nowPage; 
$arrayPagelnfo[ countPage] = $countPage; 
S$arrayPagelnfo['first] = 1; 
if ($page > 1){ 

S$arrayPagelnfo['previous'] = $rs->AbsolutePage() - 1; 
}else{ 
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$arrayPagelnfo[previous'] = 1; 


} 
if ($page < $countPage) { 

$arrayPagelnfo[ next] = $rs->AbsolutePage() + 1; 
}else{ 

S$arrayPagelnfo['next] = $countPage; 


} 
S$arrayPagelnfo[last] = $countPage; 
} 
return $arrayPagelnfo; /1 返回 包含 所 有 分 页 信息 的 数组 


由 
PageDB 类 主要 实现 ADODB 类 库 操 作 MySQL 数据 库 的 分 页 显示 ， 类 中 所 定义 的 pageData0 方 法 用 于 
返回 一 个 参数 数组 ， 该 数组 包含 所 有 分 页 信息 ， 如 每 页 显示 的 记录 数 、 总 页 数 、 总 记录 数 及 分 页 导航 链接 
具体 页 码 值 等 ， 这 样 只 需 根据 返回 数组 即 可 获取 所 有 分 页 信息 ， 为 进一步 开发 做 好 铺垫 。 
(3) 建立 StaticPage 类 用 于 动态 生成 静态 页 面 ， 该 类 主要 使 用 PHP 中 缓存 函数 实现 。 
具体 实现 代码 如 下 : 


class StaticPage 
{ 
function pageBegin () // 指 定 要 生成 静态 页 的 开始 部 分 
{ 
ob_start(); 
function pageEnd () /定义 要 生成 静态 页 的 结束 部 分 
ob_end_flush(); 
有 
function getStaticPageName ($url, $extendsName, $defaultPage = 'index') 
// 根 据 动态 页 的 实际 地 址 生成 静态 页 的 名 称 
$baseName = basename($url); /不 包含 目录 的 PHP 动态 页 名 
$pageName = substr($baseName, 0, strpos($baseName, '.php')); 。 // 去 掉 扩展 名 


if ($pageName == "){ 
$pageName = $defaultPage; 


$searchString = substr(strchr($baseName, '?'), 1, strlen(strchr($baseName, '?"))); 


/获取 查询 字符 串 部 分 

$arraySearchString = explode('&', $searchString); /分 割 查询 字符 串 并 保存 到 数组 中 
$searchName = "; 
for ($i = 0; $i < count($arraySearchString); $i ++) { /使 用 “-” 连 接 查 询 字符 串 的 值 

$searchStringTmpValue = strstr($arraySearchString[$i], '="); 

if ($searchStringTmpValue != ") { 

$searchName .= '-' . substr($searchStringTmpValue, 1, strlen($searchStringTmpValue)); 
上 
SfieName = $pageName . $searchName .… . $extendsName; /| 生成 静态 页 文件 名 
return $fileName; // 返 回 静态 页 文件 名 
function makeStaicPage ($url, $entendsName, $saveDin) 

/生成 静态 页 文件 
$content = ob_get_contents(); /获取 PHP 页 面 生成 的 HTML 页 面 内 容 


if(Lis_dir($saveDiD){ 


人 
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mkdir($saveDir); /创建 保存 静态 页 的 目录 


} 

$fileName = S$this->getStaticPageName(S$url, $entendsName); 

S$fp = fopen($saveDir ./ . $fileName, Ww'); 

fwrite($fp, $content); // 将 HTML 内 容 写 入 静态 页 
} 


function reMakeCondition ($saveDirAndFile, $reMakeTime = -1) 


// 设 定 重新 生成 静态 页 的 条 件 
if ($reMakeTime ==-1){ 
if (file_exists($saveDirAndFile)) { 
return true; 


} 

} elseif (file_exists($saveDirAndFile) && time() - filemtime($saveDirAndFile) < $reMakeTime) { 
// 如 果 静 态 页 超出 指定 的 时 间 范 围 或 不 存在 则 返回 true， 表 示 将 重新 生成 静态 页 
return true; 


} 
return false; 


} 


} 

上 述 代 码 中 定义 的 pageBegin0 和 pageEnd0 方 法 分 别 用 来 定义 所 要 生成 的 静态 页 面 的 开始 和 结束 位 置 ， 
定义 的 getStaticPageName() 方 法 用 来 根据 实际 PHP 页 面 的 请 求 地 址 生成 静态 页 面 地 址 , makeStaicPage() 方 法 
用 于 创建 静态 页 面 ， 最 后 定义 的 reMakeCondition0 方 法 用 于 指定 重新 生成 静态 页 面 的 条 件 。 

分 析 上 述 代 码 可 知 ， 本 实例 所 讲述 的 生成 静态 页 的 方法 是 真正 意义 上 的 生成 静态 页 面 ， 而 非 伪 静态 ， 
这 样 的 好 处 是 不 受 服务 器 局 限 ， 而 使 用 伪 静 态 的 方法 需要 开启 Apache 服务 器 的 Rewrite 功能 ， 如 果 用 户 使 
用 租用 的 空间 ， 可 能 有 些 空间 提供 商 未 提供 上 述 技术 支持 。 

(4) 建立 index.php 页 面 ， 用 于 实现 学 生 信息 的 分 页 显示 ， 在 页 面 最 开始 调用 StaticPage 类 ， 并 根据 
reMakeCondition0 方 法 判断 与 当前 PHP 页 面 所 对 应 的 静态 页 面 是 否 存在 或 静态 页 面 是 否 已 经 过 期 ， 如 果 是 
则 重新 生成 , 否则 使 用 headerO 函 数 重新 定向 到 与 该 页 面 所 对 应 的 静态 页 面 中 , 并 使 用 exit0 函 数 终止 该 PHP 
页 面 剩余 代码 的 执行 ， 从 而 提高 代码 执行 效率 。 


<?php 
require_once 'StaticPage.php'; // 包 含 StaticPage 类 
S$ststicPage = new StaticPage(); /对 StaticPage 进行 实例 化 
$ststicPage->pageBegin(); // 定 义 生成 静态 页 面 的 起 始 位 置 
S$dirName = '"/pages'; /定义 生成 静态 页 的 目录 
SfileName = $ststicPage->getStaticPageName($_SERVER[REQUEST_URI", 'shtml"); 
/生成 与 该 PHP 页 面 对 应 的 静态 页 的 名 称 
if ($ststicPage->reMakeCondition($dirName ./. $fileName, 60)){ ”// 判 断 是 否 需要 重新 生成 静态 页 
header(location:pages/ . $fileName); // 不 需要 则 重新 定向 到 与 该 PHP 页 面 所 对 应 的 静态 页 面 
exit(); 
虽 4 
$baseUrl = TM/19/19.17"; 
require_once 'ConnDB.php'; // 包 含 数据 库 连 接 类 
require_once 'PageDB.php'; // 包 含 分 页 类 
$connObj = new ConnDB(mysql', ‘Iocalhost', 'root', 'root', 'db_database19', false); ”// 实 例 数 据 库 连接 类 
$connlD = $connObj->getConnlD(); // 返 回 数 据 库 连接 ID 
Ys 
(5) 实例 PageDB 分 页 类 ， 通 过 for 循环 显示 学 生 信 息 。 
<?php 
SpageObj = new PageDB(); /实例 分 页 类 


$pagelnfo = $pageObj->pageData("select id,studentno ,studentname, classname，tel , email from tb_ 
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student order by id ", $connlD, 5, $_GET[page]); /获取 包含 分 页 信息 的 数组 
$arrayPageData = $pagelnfof data']; /分 页 数组 下 标 为 data 项 的 元 素 为 学 生 信息 内 容 
for ($i = 0; $i < count($arrayPageData); $i ++){ // 通 过 for 循环 显示 学 生 信 息 

?> 

<!- 显 示 学 生 信息 --> 

<?php 


(6) 显示 分 页 链接 ， 学 生 信 息 显示 页 面 根据 该 链接 参数 page 的 值 决定 显示 第 几 个 页 面 。 
<div style="width:80%; height:18px:; text-align:left; padding-top:5px"> 
共有 学 生 信 息 &nbsp;<?= $pagelnfo[countRs]?>&nbsp; 条 &nbsp; 每 页 显示 &nbsp;<?= $pagelnfo 
[pageSize] ?> &nbsp; 条 &nbsp; 第 &nbsp;<?= $pagelnfo[page] ?>&nbsp; 页 / 共 &nbsp;<?= $pagelnfo[countPage] 
> 
&nbsp; 页 &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
<a href="<?= $_SERVER['PHP_SELF]?>?page=<?= $pagelnfo[first]?>”class="a1"> 首页 </a>&nbsp; 
&nbsp; 
<a href="<?= $_SERVER[PHP_SELF]?>?page=<?= $pagelnfo[ previous]?>" class="a1"> 上 一 页 </a>&nbsp; 
&nbsp; 
<a href="<?= $_SERVER['PHP_SELF']?>?page=<?= $pagelnfo[next]?>" class="a1"> 下 一 页 </a>&nbsp; 
&nbsp; 
<a href="<?= $_SERVER[PHP_SELF']?>?page=<?= $pagelnfo[last]?>" class="a1"> 尾 页 </a> 
</div> 
(7) 关闭 数据 库 的 连接 ， 并 调用 ststicPage 类 的 makeStaicPage0 方 法 生成 静态 页 ， 最 后 调用 ststicPage 
类 的 pageEnd0 方 法 指定 生成 静态 页 面 的 结束 位 置 。 


<?php 

$connObj->closeConnlD(); /| 关闭 与 数据 库 的 连接 
S$ststicPage->makeStaicPage($_SERVER['REQUEST_URI', 'shtml'，'./pages'); /| 生成 静态 页 面 
$ststicPage->pageEnd(); /指定 生成 静态 页 面 的 结束 位 置 

?> 


运行 上 述 实例 ， 当 第 一 次 访问 学 生 信息 页 面 时 所 浏览 的 是 PHP 动态 页 面 的 内 容 ， 如 果 返 回 页 面 继续 浏 


图 19.6 第 一 次 访问 PHP 页 面 图 19.7 第 二 次 及 以 后 访问 PHP 页 面 
19.5.4 ADODB 控制 结果 集 的 存 取 方法 


例 19.18 ”通过 操作 结果 集 方法 ， 对 数据 表 中 的 行进 行 操作 ， 循 环 输出 数据 表 中 的 数据 。 同 时 应 用 一 个 
新 的 方法 ， 实 现 对 结果 集 存 取 方式 的 控制 。〔 实 例 位 置 光盘 \TM\Instance\19\19.18) 
实现 步骤 如 下 。 
(1) 创建 conmn 文件 夹 ， 编 写 conn php 文件 ， 载 入 adodb.inc .php 文件 。 实 现 与 MySQL 数据 库 服 务 器 
中 db_database19 数据 库 的 连接 , 应 用 SADODB FETCH MODE 设置 结果 集 以 字段 名 称 为 索引 进行 存 取 , 并 
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设置 数据 库 的 编码 格式 为 utf-8。 


代码 如 下 : 

<?php 

include_once ("../adodb5/adodb.inc.php’); 
$conn = ADONewConnection('mysql"); 


$conn -> PConnect('localhost',root,'111',"db_database19'"); 
$ADODB_FETCH_MODE = ADODB_FETCH_ASSOC:; 


$conn -> execute('set names utf8"); 
Ws 


// 载 入 adodb.inc.php 文 件 
/连接 MySQL 数据 库 

/连接 db_database19 数据 库 
/设置 结果 集 以 字段 名 称 为 索引 
/设置 编码 格式 


(2) 创建 index.php 文件 ， 完 成 数据 的 循环 输出 。 首 先 ， 包 含 数据 库 连接 文件 。 然 后 ， 定 义 SQL 查询 


关键 代码 如 下 : 
<?php 
include_once 'conn/conn.php'; 


S$sqlstr = 'select * from tb_user where id limit 5'; 


Srst = $conn->execute ( S$sqlstr ) or die ( 'error: '. $conn->errorMsg () ); 


while ( ! Srst->EOF ) { //wihle 语句 循环 输出 结果 


<tr> 


<td width="126" height="30" align="center"> 


echo $rst->fields [user]; 
</td> 


<td width="117" align="center"><?php 


echo S$rst->fields [dates']; 
?> 
</td> 
</tr> 
<?php 


Srst->movenext (); 
Srst->close (); 


$conn->close (); 
?> 


运行 上 述 代 码 ， 结 果 如 图 19.8 所 示 。 


图 19.8 


控制 结果 集 存 取 方 法 的 运行 结果 


语句 ， 通 过 execute0 函 数 执 行 查询 操作 。 最 后 ， 通 过 while 语句 循环 输出 查询 结果 。 


// 载 入 数据 库 链 接 文件 
/1SQL 查询 语句 
/| 执行 查询 语句 


/指针 下 移 
/关闭 连接 
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19.6 小 结 


本 章 介 绍 了 ADODB 类 库 的 概念 、 支 持 的 数据 库 和 下 载 安装 的 方法 ， 对 类 库 中 的 常用 函数 还 作 了 详细 
的 讲解 ， 并 且 通 过 实例 来 帮助 读者 加 深 理解 。 相 信 读 者 通过 本 章 的 学 习 ， 能 够 熟练 地 应 用 ADODB 类 库 操 
作 各 种 数据 库 。 


19.7 学 习 成 果 检 验 


1. 使 用 ADODB 类 库 读 取 Access 数据 库 中 的 数据 。〔 实 例 位 置 ， 光盘 \TMNInstance\19\19.19) 
2. 使 用 ADODB 类 库 同 时 连接 多 个 数据 库 ， 并 将 Access 数据 库 表 中 的 数据 存储 到 MySQL 数据 库 中 。 
《实例 位 置 ， 光盘 \TM\Instance\19\19.20) 


#2 Os 


数据 库 编 程 技术 


( 代 视频 讲解 : 136 分 钟 ) 


PHP 所 支持 的 数据 库 类 型 较 多 , 在 这 些 数 据 库 中 ,， MySQL 数据 库 
与 PHP 的 兼容 最 好 ， 与 Linux 系统 、Apache 服务 器 和 PHP 语言 构成 
了 当今 主流 的 LAMP 网 站 架构 模式 ,并 且 PHP 提供 了 多 种 操作 MySQL 
数据 库 的 方式 ， 可 以 适合 不 同 需求 和 不 同类 型 的 项 目 。 本 章 将 介绍 如 
何 通过 MySQL 酌 数 据 作 数 据 库 ， 这 也 是 中 小 型 项 目 常 用 的 方式 之 一 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

Wp 了 解 PHP 访问 MySQL 数据 库 的 步骤 

由 掌握 常用 MySQL 数据 库 困 数 的 使 用 方法 

Hi 掌握 PHP 操作 MySQL 数据 库 进 行 增 、 删 、 改 、 查 的 操作 

掌握 数据 分 页 显示 的 实现 方法 

WI 掌握 应 用 多 种 方法 获取 结果 集 

Hi 掌握 获取 查询 结果 集中 的 记录 行 数 
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20.1 PHP 访问 MySQL 数据 库 的 一 般 步 又 


和 其 他 语言 类 似 ，PHP 操作 MySQL 数据 库 的 


过 程 一 般 分 为 5 步 ， 分 别 为 连接 MySQL 数据 库 服 Ee ! 
务 器 、 选 择 数据 库 、 执行 SQL 语句 、 关 闭 结果 集 以 二 Ee 
及 电 开 与 MySQL 服务 器 的 连接 ， 如 图 20.1 所 示 。 竺 | 入 
下 面 将 具体 介绍 其 实现 过 程 。 执行 sq1 通 名 ! 
1. 连接 MySQL 服务 器 关闭 结果 集 | 请 各 
应 用 mysql_connect0 函 数 建立 与 MySQL 服务 i | 


器 的 连接 , 并 返回 一 个 连接 标识 , 在 以 后 对 MySQL 
服务 器 进行 操作 时 ， 可 以 根据 这 个 连接 标识 定位 不 
同 的 连接 。 
2. 选择 数据 库 
应 用 mysql_select_db0 函 数 选择 MySQL 数据 库 服 务 器 上 的 数据 库 ， 并 与 该 数据 库 建 立 连接 。 
3. 执行 SQL 语句 
在 选择 的 数据 库 中 应 用 mysql_query0 函 数 执行 SQL 语句 。 对 数据 的 操作 主要 包括 以 下 5 种 方式 。 
查询 数据 : 应 用 select 语句 实现 数据 的 查询 功能 。 
显示 数据 : 应 用 select 语句 显示 数据 的 查询 结果 。 
插入 数据 : 应 用 insert 语句 向 数据 库 中 插入 数据 。 
更 新 数据 :应 用 update 语句 修改 数据 库 中 的 记录 。 
删除 数据 : 应 用 delete 语句 删除 数据 库 中 的 记录 。 
.关闭 结果 集 
数据 库 操作 完成 后 ， 需 要 关闭 结果 集 ， 以 释放 系统 资源 。 


mysql_free_result($result); 


图 20.1 PHP 访问 MySQL 数据 库 的 一 般 步 又 


因 罗 办 罗 加 


a 


[a 技巧 如 果 在 多 个 网 页 中 都 要 频繁 进行 数据 库 访问 , 则 可 以 建立 与 数据 库 服 务 器 的 持续 连接 来 提高 效 
率 。 因为 每 次 与 数据 库 服 务 器 的 连接 需要 较 长 的 时 间 和 较 大 的 资源 开销 , 持续 的 连接 相对 来 说 会 更 有 效 。 
建立 持续 连接 的 方法 就 是 在 数据 库 连 接 时 , 调用 mysql pconnect0 〇 函数 代替 mysql_connect(O 函 数 。 建 立 的 
持续 连接 在 本 程序 结束 时 ， 不 需要 调用 mysql_close() 函 数 来 关闭 。 下 次 程序 在 此 执行 mysql_pconnect() 
阴 数 时 ， 系 统 自动 直接 返回 已 经 建立 的 持续 连接 的 ID 号 ， 而 不 再 去 真 的 连接 数据 库 。 


5. 断 开 与 MySQL 服务 器 的 连接 

每 使 用 一 次 mysql_connect0 或 mysql_query0 函 数 ， 都 会 消耗 系统 资源 。 这 在 少量 用 户 访问 Web 网 站 
时 影响 不 明显 ， 但 如 果 用 户 连接 超过 一 定数 量 ， 就 会 造成 系统 性 能 的 下 降 ， 甚 至 死机 。 为 了 避免 这 种 现象 
的 发 生 ， 在 完成 数据 库 的 操作 后 ， 可 以 应 用 mysql_close0 函 数 关闭 与 MySQL 服务 器 的 连接 ， 以 节省 系统 
资源 。 

mysql_close(Slink): 


@ 
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20.2 ”PHP 操作 MySQL 数据 库 的 方法 


20.1 节 简 要 介绍 了 PHP 访问 和 操作 MySQL 数据 库 的 一 般 步 骤 .PHP 提供 了 大 量 操作 MySQL 数据 库 的 
方式 ， 其 中 函数 方式 相对 简便 并 且 应 用 较 广 泛 。 本 节 将 详细 介绍 PHP 中 MySQL 相关 函数 的 使 用 方法 。 


20.2.1 使 用 mysql_connect() 函 数 连接 MySQL 服务 器 


PHP 操作 MySQL 数据 库 ， 首 先 要 建立 与 MySQL 数据 库 的 连接 ，PHP 实现 与 数据 库 的 连接 相对 简便 ， 
只 需 使 用 mysql_connectO 函 数 即 可。 函数 语法 如 下 : 

resource mysql_connect ( [string server [, string username [, string password [, bool new_link [, int client_flags]]]]] ) 

Imysql_connectO 函 数 用 于 打开 一 个 到 MySQL 服务 器 的 连接 ， 如 果 成 功 则 返回 一 个 MySQL 连接 标识 ， 
失败 则 返回 false。 该 函数 的 参数 如 表 20.1 所 示 。 


表 20.1 mysql_connect() 函 数 的 参数 说 明 


参数 说 明 
MySQL 服务 器 。 可 以 包括 端口 号 ， 如 "hostname:port"; 或 者 到 本 地 套 接 字 的 路 径 ， 如 对 于 localhost 的 


SeIVeT ， 


"“/path/to/socket"。 如 果 PHP 指令 mysql.default_ host 未 定义 (默认 情况 )， 则 默认 值 是 'localhost:3306 
Usemame | 用 户 名 。 默 认 值 是 服务 器 进程 所 有 者 的 用 户 名 


assword 密码 。 默 认 值 是 空 密码 

如 果 用 同样 的 参数 再 次 调用 mysql_connect0 函 数 , 将 不 会 建立 新 连接 , 而 将 返回 已 经 打开 的 连接 标识 。 
new_link 参数 new_link 改变 此 行为 并 使 mysql_connect0 函 数 总 是 打开 新 的 连接 , 即使 mysql_connectO 函 数 曾 在 
前 面 被 用 同样 的 参数 调用 过 

, client flags 参数 可 以 是 以 下 常量 的 组 合 : MYSQL CLIENT SSL, MYSQL CLIENT COMPRESS， 
Client flags | MYSQL CLIENT IGNORE SPACE 或 MYSQL CLIENT INTERACTIVE 


例 20.1 应 用 mysql_connect0 函 数 创建 与 MySQL 服务 器 的 连接 ，MySQL 数据 库 服 务 器 地 址 为 
127.0.0.1， 用 户 名 为 root， 密 码 为 111。 (实例 位 置 ;， 光盘 \TM\Instance\20\20.1) 


代码 如 下 : 

<?php 

$host = "127.0.0.1"; /MySQL 服务 器 地 址 
$userName = "root"; lI 用 户 名 

S$password = "111"; /密码 


if($connID = mysql_connect($host, $userName, $password)}{ 
// 建 立 与 MySQL 数据 库 的 连接 ， 并 弹出 提示 对 话 框 
echo "<scriptlanguage='javascript>alert( 数 据 库 连 接 成 功 ! ');</script>"; 
Jelse{ 
echo "<script language='ijavascript>alert(' 数 据 库 连 接 失败 ! ');</script>"; 
运行 上 述 代 码 ， 如 果 在 本 地 计算 机 中 安装 了 MySQL 数据 库 ， 并 且 root 用 户 
名 为 root， 密 码 为 111， 则 会 弹出 如 图 20.2 所 示 的 对 话 框 。 


> 
要 技 现 在 项 目 完成 后 ， 为 了 屏蔽 由 于 数据 库 连接 失败 而 显示 的 不 友好 的 
错误 信息 , 可 以 在 mysql connect0 浮 数 前 加 “@”, 该 符号 用 来 屏蔽 错误 提示 。 20.2 数据库 连 接 成 功 


@ 
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20.2.2 ”使 用 mysql_select_db() 函 数 选择 数据 库 文件 


成 功 与 MySQL 数据 库 建 立 连接 后 ， 需 要 选择 MySQL 数据 库 服 务 器 中 指定 的 数据 库 。PHP 中 使 用 
mysql_select_db0 函 数 实现 数据 库 的 选择 功能 。 该 函数 的 语法 格式 如 下 : 

bool mysql_select db ( string database_name [, resource link_identifier] ) 

Imysql_select_ db0 函 数 用 于 设 定 与 指定 的 连接 标识 符 所 关联 的 服务 器 上 的 当前 激活 数据 库 。 如 果 没 有 指 
定 连 接 标 识 符 ， 则 使 用 上 一 个 打开 的 连接 。 如 果 没有 打开 的 连接 ， 本 函数 将 无 参数 调用 mysql_connectO 函 数 
来 尝试 打开 一 个 使 用 。 其 后 的 每 个 mysql_query0 函 数 调用 都 会 作用 于 当前 激活 的 数据 库 。 该 函数 的 参数 说 
明 如 表 20.2 所 示 。 


表 20.2 ”mysql_select_db() 函 数 的 参数 说 明 


参数 说 有明 
database_name 必 选 参数 ， 用 户 指定 要 选择 的 数据 库 名 称 
link identifier 可 选 参数 ， 数 据 库 连 接 ID， 如 果 省 略 该 参数 ， 则 默认 为 最 近 一 次 与 数据 库 建立 的 连接 


例 20.2 首先 使 用 mysql_connectO 函 数 建立 与 MySQL 数据 库 的 连接 并 返回 数据 库 连 接 ID， 然 后 使 用 
mysql_ select_ db0 函 数 选择 MySQL 数据 库 服 务 器 中 名 为 db_database20 的 数据 库 。〈 实 例 位 置 ， 光盘 \IM\ 
Instance\20\20.2) 


实现 代码 如 下 : 
<?php 
$host = "127.0.0.1"; /MySQL 服务 器 地 址 
S$userName = "root"; /| 用 户 名 
S$password = "111"; /密码 
$dbName = "db_database20"; /数据 库 
$connlD = mysql_connect($host, $userName, $password); /建立 与 MySQL 数据 库 服务 器 的 连接 
if(mysql_select_db($dbName, $connID)X{ /| 选择 数据 库 
echo "数据 库 选择 成 功 !"; 
jelsef 
echo "数据 库 选择 失败 ! "; 
四 


运行 上 述 代码 ， 如 果 本 地 MySQL 数据 库 服务 器 中 存在 名 为 db_database20 的 数据 库 ， 将 在 页 
如 图 20.3 所 示 的 提示 信息 。 


冯 
局 
+ 
咎 
dl 


20.2.3 ”使 用 mysql_query() 函 数 执行 SQL 语句 


成 功 选择 MySQL 数据 库 服务 器 中 的 数据 库 后 ， 即 可 对 所 选 数据 库 中 的 数据 表 进 行 查询 、 更 改 以 及 删除 
等 操作 ，PHP 使 用 mysql_queryO 函 数 就 可 以 实现 上 述 所 有 操作 ， 操 作 极 其 简便 ， 说 明 在 PHP 底层 进行 了 复 
杂 的 封装 ， 而 提供 给 上 层 开 发 人 员 一 种 简便 的 编程 模式 ， 这 也 是 PHP 操作 简便 的 体现 和 应 用 广泛 的 原因 。 
mysql_query0 函 数 的 语法 格式 如 下 : 

resource mysql_query ( string query [, resource link_identifier] ) 


mysql_query0 函 数 用 于 执行 一 条 查询 语句 ， 该 函数 的 参数 说 明 如 表 20.3 所 示 。 


全 
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表 20.3 mysql_query() 函 数 的 参数 说 明 


参数 说 明 
query 字符 串 类 型 ， 传 入 的 是 SQL 的 指令 
i 资源 类 型 ， 传 入 的 是 由 mysql_connect0 函 数 或 mysql_pconnect0 函 数 返回 的 连接 号 。 如 果 省 


略 该 参数 ， 则 会 使 用 最 后 一 个 打开 的 MySQL 数据 库 连接 
例 20.3 ”查询 学 生 信息 表 中 学 生 的 成 绩 信 息 。 (实例 位 置 ， 光盘 \TMNInstance\20\20.3) 


代码 如 下 : 
<?php 
$host = "127.0.0.1"; /MySQL 数据 库 服务 器 
S$userName = "root"; /用 户 名 
$password = "111"; /密码 
$dbName = "db_database20"; /| 数据库 名 
$connlD = mysql_connect($host, $userName, $password); // 连 接 MySQL 数据 库 
mysql_select_db($dbName, $connID); /| 选择 MySQL 数据 库 
mysql_query("set names utf8"); /设置 字符 集 
echo "<table border=\"1px\" align=\"center\"> 
<tr> 
<td> 学 号 </td> 
<td> 姓 名 </td> 
<td> 班 级 </td> 
<td> 语 文 </td> 
<td> 数 学 </td> 
<td> 英 语 </td> 
< 
$query = mysql_query("select sno, sname, class, chinese, math ,english from tb_student", $connID); 
// 执 行 查询 
while($result = mysql_fetch_array($query)) // 获 取 结 果 集 并 输出 查询 结果 
上 
echo “<tr> 
<td>".$result["sno"]."</td> 
<td>".$result["sname"]."</td> 
<td>".$result["class"]."</td> 
<td>".$result["chinese"]."</td> 
<td>".$result["math"]."</td> 
<td>".$result["english"]."</td> 
</tr>"; 
} 
echo "</table>"; 
Ta 


运行 上 述 实例 ， 结 果 如 图 20.4 所 示 。 


曾 hapyochessaaa61283/ndexphp _ Windows intemet Explorer [eld] 
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图 20.3 ”数据库 选择 成 功 图 20.4 查询 学 生成 绩 
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20.2.4 应 用 mysql_fetch_array() 函 数 从 数组 结果 集中 获取 信息 


视频 讲解 : 光盘 \TM\Video\ 第 20 章 \ 应 用 mysql_fetch_array0 函 数 从 数组 结果 集中 获取 信息 .exe 
使 用 mysql_query0 函 数 执行 查询 后 ， 并 不 能 返回 查询 结果 ， 那 么 如 何 才能 获取 查询 结果 呢 ? PHP 提供 
了 mysql_fetch_array0) 函 数 实现 获取 查询 结果 集 的 功能 。 该 函数 的 语法 格式 如 下 。 
array mysql_fetch_array ( resource result [, int result_ type] ) 
mysql fetch_arrayO 函 数 的 参数 说 明 如 表 20.4 所 示 。 
表 20.4 mysql_fetch_array() 函 数 的 参数 说 明 
参数 说 明 
result 资源 类 型 的 参数 ， 要 传 入 的 是 由 mysql_query0 函 数 返回 的 数据 指针 


可 选项 , 整数 型 参数 , 要 传 入 的 是 MYSQL_ASSOC、MYSQL NUM、MYSQL BOTH 3 种 由 PHP 
定义 好 的 常数 之 一 ， 默 认 值 是 MYSQL_BOTH 
result_type 用 MYSQL ASSOC 只 得 到 关联 索引 (相当 于 mysql fetch_assoc0 函 数 ) 
用 MYSQL NUM 只 得 到 数字 索引 (相当 于 mysql_fetch row0 函 数 ) 
用 MYSQL BOTH 将 得 到 一 个 同时 包含 关联 和 数字 索引 的 数组 


例 20.4 按 员 工 编号 以 模糊 查询 的 方式 查询 员工 信息 ， 并 显示 全 部 查询 结果 。 《实例 位 置 ， 光盘 \TM 
Instance\20\20.4) 

具体 实现 步 又 如 下 所 示 。 

(1) 建立 与 MySQL 数据 库 的 连接 ， 并 返回 数据 库 连接 ID。 其 代码 如 下 : 


<?php 

$connID=mysql_connect("localhost","root","111"); // 建 立 与 数据 库 的 连接 
mysql_select_db("db_database20", $connID); /| 选择 数据 库 
mysql_query("set names utf8"); /设置 字符 集 

?> 


(2) 建立 查询 信息 录入 表单 ， 表 单 及 表单 元 素 如 表 20.5 所 示 。 
表 20.5 员工 信息 录入 表单 


说 了 明 
表单 
name="number" type="text" id="number" 录入 员工 编号 
type="hidden" name="flag" value="1" 判断 表单 是 否 提交 
name="submit" type="submit" value=" 提 交 " 提交 按钮 


(3) 使 用 $_POST 全 局 数组 接收 表单 提交 的 flag 元 素 的 值 ， 并 使 用 issetO 函 数 判断 是 否 已 经 设置 了 该 元 
素 的 值 ， 如 果 已 设置 则 说 明 已 经 提交 了 表单 ， 然 后 采用 模糊 查询 的 方式 查询 所 有 与 查询 关键 字 相 匹配 的 员 
工 信 息 ， 并 使 用 while 循环 将 查询 结果 显示 出 来 。 其 代码 如 下 : 

<?php 


if(isset($_POST["flag"])X{ 
$query=mysql_query("select * from tb_employee where number like '%".$_POST["number"]."%"); 


if($query{ 
while($myrow=mysql_fetch_array($query)){ 


?> 


@ 
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<td align="center” bgcolor="#FFFFFF" class="STYLE4"><span class="STYLE2"><?php echo $myrow 
[number];? > </span></td> 
<td align="center" bgcolor="#FFFFFF" class="STYLE4"><span class="STYLE2"><?php echo $myrow 
[name];?> </span></td> 
<td height="23" align="center" bgcolor="#FFFFFF" class="STYLE4"><span class="STYLE2"><?php echo 
$myrow [tel];?></span></td> 
<td height="23" align="center" bgcolor="#FFFFFF" class="STYLE4"><span class="STYLE2"><?php echo 
$myrow [address];?></span></td> 
</tr> 
<?php 


b 
» 


2 
运行 该 实例 ， 在 员工 查询 信息 录入 表单 中 输入 员工 编号 ， 然 后 单 击 “ 提 交 ” 按 钮 ， 即 可 以 模糊 查询 的 
方式 查询 出 所 有 与 查询 关键 字 相 匹 配 的 员工 信息 ， 如 图 20.5 所 示 。 
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查询 企业 员工 的 详细 信息 
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图 20.5 查询 员工 信息 


20.2.5 ”应 用 mysql_fetch_object() 函 数 从 结果 集中 获取 一 行 作为 对 象 


顽 1 视频 讲解 : 光盘 \TM\Video\ 第 20 章 \ 应 用 mysql_fetch_objectO 函 数 从 结果 集中 获取 一 行 作 为 对 象 .exe 

20.2.4 节 中 讲解 了 应 用 mysql_fetch_array0 函 数 来 获取 结果 集中 的 数据 。 除 了 这 个 方法 以 外 ， 应 用 
mysql_fetch_objectO 函 数 也 可 以 轻松 实现 这 一 功能 ， 下 面 通过 同一 个 实例 的 不 同方 法 来 体验 一 下 这 两 个 函数 
在 使 用 上 的 区 别 。 首 先 介绍 mysql_fetch_objectO 函 数 。 

语法 如 下 : 

object mysql_fetch_object ( resource result ) 

mysql_fetch_object0 函 数 和 mysql_fetch_array0 函 数 类 似 , 只 有 一 点 区 别 :前 者 返回 一 个 对 象 而 不 是 数组 ， 
即 该 函数 只 能 通过 字段 名 来 访问 数组 。 访 问 结果 集中 行 元 素 的 语法 结构 如 下 : 

$row->col_name /icol_name 为 列 名 ，$row 代表 结果 集 

例如 ， 如 果 从 某 数据 表 中 检索 id 和 name 值 ， 可 以 用 $row->id 和 $row-> name 访问 行 中 的 元 素 值 。 


人 注意 本 函数 返回 的 字段 名 是 区 分 大 小 写 的 ， 这 是 初学 者 学 习 时 最 容易 忽视 的 问题 。 


例 20.5 使 用 mysql fetch_object0 函 数 获取 查询 到 的 图 书信 息 。( 实 例 位 置 : 光盘 \TMNInstance\20\20.S) 
具体 开发 步骤 如 下 。 


@ 
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(1) 建立 图 书 查询 表单 ， 表 单 及 表单 元 素 说 明 如 表 20.6 所 示 。 
表 20.6 图 书信 息 查询 表单 及 表单 说 明 


name="myform" method="post" action="" 
name="txt_book" type="text" id="txt_book" size="25" 
type="submit" name="Submit" value=" 查 询 " 


(2) 建立 与 MySQL 数据 库 的 连接 ， 设 置 字符 集 ， 并 返回 数据 库 连 接 DD。 其 代码 如 下 : 
$link=mysql_connect("localhost","root","111") or die(" 数 据 库 连 接 失 败 ".mysql_error()); 。“”// 建 立 与 数据 库 的 连接 
mysql_select_db("db_database20",$link); // 选 择 数 据 库 


mysql_query("set names gb2312"); // 设 置 字符 集 
(3) 应 用 mysql_ query0 函 数 执行 SQL 查询 语句 ， 并 使 用 mysql_fetch_objectO 函 数 获取 查询 语句 的 结果 
集 。 代 码 如 下 : 
if ($_POST[Subm==" 查 询 "){ 
Stxt_book=$_POSTI[txt_book]: // 接 收 查询 关键 字 


ee query("select* from tb_book where bookname like '%".trim($txt_book)."%"); 
选择 的 条 件 为 “like”， 则 进行 模糊 查询 
es fetch_object($sql); 


} 
(4) 应 用 do…while 循环 语句 以 对 象 的 方式 输出 结果 集中 的 图 书信 息 到 浏览 器 中 。 其 代码 如 下 : 
d 
2 
<tr align="left" bgcolor="#FFFFFF"> 
<td height="20" align="center"><?php echo $info->id; ?></td> 
<td >&nbsp;<?php echo $info->bookname; ?></td> 
<td align="center"><?php echo $info->issuDate; ?></td> 
<td align="center"><?php echo $info->price; ?></td> 
<td align="center">&nbsp;<?php echo $info->maker; ?></td> 
<td>&nbsp;<?php echo $info->publisher; ?></td> 
</tr> 
<?php 
}while($info=mysql_fetch_object($sql)); 
保存 index.php 动态 页 ， 在 正 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 运 行 结果 如 图 20.6 所 示 。 


20.2.6 应 用 mysql_fetch_row() 函 数 逐 行 获取 结果 集中 的 每 条 记录 


名 视频 讲解 : 光盘 \TMIVVideo\ 第 20 章 \ 应 用 mysql_fetch_row0 函 数 逐 行 获取 结果 集中 的 每 条 记录 .exe 

前 面 介 绍 了 应 用 mysql_fetch_array0 和 mysql_fetch_objectO 函 数 来 获取 结果 集中 的 数据 。 本 节 向 读者 介绍 
第 3 种 方法 ， 应 用 mysql_fetch row0 函 数 逐 行 获取 结果 集中 的 每 条 记录 。 首 先 来 了 解 mysql_fetch rowO 函 数 。 

语法 如 下 : 

array mysql_fetch_row ( resource result ) 

mysql_query0 函 数 将 查询 语句 发 送 到 服务 器 中 执行 。 查询 语句 不 使 用 分 号 终止 。 如 果 查 询 非法 或 由 于 某 
些 原因 不 能 执行 ， 则 mysql_query0 函 数 返 回 false， 和 否则 返回 一 个 结果 集 标 识 符 。mysql_fetch row0 函 数 从 和 
指定 的 结果 标识 关联 的 结果 集中 获取 一 行 数据 并 作为 数组 返回 ， 将 此 行 赋 给 变量 Srow， 每 个 结果 的 列 储存 
在 一 个 数组 的 单元 中 ， 偏 移 量 从 0 开始 ， 即 以 Srow[0] 的 形式 访问 第 一 个 元 素 (只 有 一 个 元 素 时 也 是 如 此 〉。 
依次 调用 mysql_fetch_row0 函 数 将 返回 结果 集中 的 下 一 行 ， 直 到 没有 更 多 行 则 返回 false。 

例 20.6 查询 图 书信 息 , 并 使 用 mysql fetch row0 函 数 获取 结果 集 显 示 图 书信 息 .( 实 例 位 置 : 光盘 \IM\ 
Instance\20\20.6) 
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有 具体 开发 步骤 如 下 。 
(1) 创建 项 目 、 添 加 表单 、 连 接 MySQL 服务 器 以 及 设置 默认 数据 库 的 实现 过 程 与 例 20.5 开发 步骤 中 
的 (1) 一 (2) 相同 ， 这 里 就 不 再 资 述 了 。 
(2) 在 应 用 mysql query0 函数 执行 SQL 查询 语句 后 ， 与 实例 20.5 不 同 的 是 ， 本 实例 使 用 
mysql fetch rowO 函 数 获取 查询 语句 的 结果 集 。 其 代码 如 下 : 
<?php 
$sql=mysql_query("select * from tb_book"); 
$row=mysql_fetch_row ($sq)l); 
if ($_POST[Submit]==" 查 询 "X{ 
S$txt_book=$_POSTI[txt_book]; 
// 如 果 选 择 的 条 件 为 “like”， 则 进行 模糊 查询 
$sql=mysql_query("select * from tb_book where bookname like '%". trim($txt_book)."%"); 
$row=mysql_fetch_row($sql); 
加 
(3) 应 用 让 条 件 语句 对 结果 集 变量 $info 进行 判断 ， 如 果 该 值 为 假 ， 则 检索 的 图 书信 息 不 存在 ， 应 用 
echo 语句 输出 提示 信息 。 其 代码 如 下 : 
<?php 


if($row==falseX{ // 如 果 检 索 的 信息 不 存在 ， 则 输出 相应 的 提示 信息 
echo "<div align='center style='color#FF0000; font-size:12px'> 对 不 起 ， 您 检索 的 图 书信 息 不 存在 !</div>"; 
， 
(4) 应 用 do…while 循环 语句 以 对 象 的 方式 输出 结果 集中 的 图 书信 息 到 浏览 器 中 。 其 代码 如 下 : 
<?php 
dof 
?> 


<tr align="left" bgcolor="#FFFFFF"> 
<td height="20" align="center"><?php echo $row[0]; ?></td> 
<td >&nbsp;<?php echo $row[1]; ?></td> 
<td align="center"><?php echo $row[2]; ?></td> 
<td align="center"><?php echo $row[3]; ?></td> 
<td align="center">&nbsp;<?php echo $row[4]; ?></td> 
<td>&nbsp;<?php echo $row[5]; ?></td> 

</tr> 

<?php 

}while($row=mysql_fetch_row($sql)); 

?> 


保存 index.php 动态 页 ， 在 正 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 运 行 结果 如 图 20.7 所 示 。 


ES 
_ hopsocdhst oa NG TE 55|xjP sess pp- 
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[ My = 2 全 医 -|« 乱 记 有- x EF EE TE 
本 | 
| ,EE En [可 TCR EE | 
| PE 条 半 辆 程 自 字 手 几 如 十 日 科技 “| 人 民利 让 出 频 社 
3 Er LT 不 再 本 有 ES 机 | 本 和 搞 | 在于 
CD ET | | | AR 
”| ?了 打扫 压条 统 开 避 完 全 放 用 2-06-12 明日 科技 “| 人 民 村 电 册 版 社 和 全 加 吉日 科技 “| 人 民 册 让 出 版 社 
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本地 Intraner | 是 六 得 式 : 委 用 区 扩 l00% > 入 下 地 lntranet | 开 P 锭 式 : 要 用 而 成 100% ~ 
图 20.6 查询 图 书信 息 图 20.7 使 用 mysql ftch row0 函 数 获取 结果 集 查 询 图 书信 息 
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20.2.7 ”应 用 mysql_num_rows() 函 数 获取 查询 结果 集中 的 记录 数 


本 视频 讲解 : 光盘 \TM\Video\ 第 20 章 \ 应 用 mysql_ num_rows(0 函 数 获 取 查 询 结果 集中 的 记录 数 .exe 

要 获取 由 select 语句 查询 到 的 结果 集中 行 的 数目 ， 必 须 使 用 mysql_num_ rows0 函 数 实现 。 首先 来 看 一 下 
该 函数 的 语法 结构 。 

语法 如 下 : 


int mysql_num_rows ( resource result ) 


ot 意 使 用 mysql unbuffered queryO 函 数 查询 到 的 数据 结果 ， 无 法 使 用 mysql num rows() 函 数 来 获 
取 查 询 结果 的 记录 数 。 


例 20.7 使 用 mysql num rows0 函 数 获取 查询 结果 的 记录 数 。( 实 例 位 置 : 光盘 \TM\Instance\20\20.7) 
具体 开发 步骤 如 下 。 

(1) 建立 与 MySQL 数据 库 的 连接 ， 设 置 字符 集 为 GB2312， 并 返回 数据 库 连 接 ID。 其 代码 如 下 ; 
$link=mysql_connect("localhost","root","111") or die(" 数 据 库 连 接 失 败 ".mysql_error()); /建立 连接 


mysql_select_db("db_database20",$link); /| 选择 数据 库 
mysql_query("set names gb2312"); // 设 置 字符 集 

(2) 默认 情况 下 显示 所 有 图 书信 息 。 其 代码 如 下 : 

S$sql=mysql_query("select * from tb_book"); /查询 所 有 图 书信 息 
$info=mysql_fetch_object($sql); // 获 取 结 果 集 


(3) 如 果 用 户 输入 了 查询 关键 字 ， 并 单 击 了 “查询 ”按钮 ， 则 采用 模糊 查询 的 方式 显示 出 所 有 符合 条 
件 的 记录 。 其 代码 如 下 : 
if ($_POSTISubmit==" 查 询 "X{ 
S$txt_book=$_POSTItxt_book]; 
$sql=mysql_query("select * from tp_book where bookname like '%".trim($txt_book)."%"); 
// 如 果 选择 的 条 件 为 “like”， 则 进行 模糊 查询 
$info=mysql_fetch_object($sql); 


} 

if($info==falseX{ // 如 果 检 索 的 信息 不 存在 ， 则 输出 相应 的 提示 信息 
echo "<div align='center style='color:#FF0000; font-size:12px'> 对 不 起 ， 您 检索 的 图 书信 息 不 存在 !</div>"; 

» 


dof 
> 
<tr align="left" bgcolor="#FFFFFF"> 
<td height="20" align="center"><?php echo S$info->id; ?></td> 
<td >&nbsp;<?php echo $info->bookname; ?></td> 
<td align="center"><?php echo S$info->issuDate; ?></td> 
<td align="center"><?php echo S$info->price; ?></td> 
<td align="center">&nbsp;<?php echo $info->maker ?></td> 
<td>&nbsp;<?php echo $info->publisher; ?></td> 
</tr> 
<?php 
}while($info=mysql_fetch_object($sql)); 
?> 
在 正 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 程 序 默认 输出 图 书信 息 表 中 的 全 部 图 书信 息 ， 并 自动 汇总 记录 
条 数 ， 如 图 20.8 所 示 。 在 文本 框 中 输入 要 检索 的 图 书 名 称 ， 如 “开发 ”支持 模糊 查询 ， 程 序 自动 去 除 查 
询 关 键 字 的 左右 空格 ) ， 单 击 “ 查 询 ” 按 钮 ， 即 可 按 条 件 检索 指定 的 图 书信 息 到 浏览 器 ， 并 自动 汇总 检索 


393 
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到 的 记录 条 数 ， 运 行 结果 如 图 20.9 所 示 。 


EEE 
| Ml | | EN | /AR 


ee | | 
[ 3 lli | 名 | 用 和 ee 日。 | 1 避 雪 掺 永和 发 完全 竹 月 2012-05-l2 | 受 用 E 和 后 人 网 所 二 本 
| 在 歌 认 页 应 用 mysql_mm rows0 | 应 用 mysql_num rows0 函 数 获 
函数 获取 数据 表 的 总 记录 数 [EE 取 查 询 结 果 集 中 的 总 记录 数 ETPFEEE3| 


20.8 默认 统计 数据 表 中 所 有 记录 图 20.9 应 用 mysql_num rowsO 函 数 获 取 查询 结果 集中 的 记录 数 


p 4 


入 意 如 果 要 获取 由 insert、 update、delete 语句 所 影响 到 的 数据 行 数 , 必须 应 用 mysql affected rows() 
函数 来 实现 。 


20.2.8 关闭 连接 


铬 gl 视频 讲解 : 光盘 \TM\Video\ 第 20 章 \ 关 闭 连 接 .exe 

完成 对 数据 库 的 操作 后 ， 需 要 及 时 断 开 与 数据 库 的 连接 并 释放 内 存 ， 否 则 会 浪费 大 量 的 内 存 空间 ， 在 
访问 量 较 大 的 Web 项 目 中 ， 很 可 能 导致 服务 器 崩溃 。 在 MySQL 函数 库 中 ， 使 用 mysql_close0 函 数 断 开 与 
MySQL 服务 器 的 连接 。 该 函数 的 语法 格式 如 下 : 

bool mysql_close ( [resource link_identifier] ) 

mysql_close0 函 数 用 于 关闭 指定 的 连接 标识 所 关联 的 MySQL 服务 器 的 非 持久 连接 。 如 果 没 有 指定 
link_identifier 参数 ， 则 关闭 最 后 一 个 打开 的 连接 。 

例 20.8 使 用 mysql_connect0 函 数 建立 与 数据 库 的 连接 ， 然 后 使 用 mysql_select_db0 函 数 选 择 数 据 库 并 
使 用 mysql_close0 函 数 断 开 与 MySQL 数据 库 的 连接 ， 在 断 开 与 MySQL 数据 库 连 接 后 ， 再 次 使 用 mysql_ 
select_db0 函 数 选择 数据 库 ， 从 两 次 选择 数据 库 的 情况 来 判断 mysql_close0 函 数 是 否 能 起 到 断 开 数 据 库 连 接 
的 作用 。 《实例 位 置 : 光盘 \TMNInstance\20\20.8) 


实例 代码 如 下 : 

<?php 

$host = "127.0.0.1"; /MySQL 数据 库 服务 器 
$userName = "root"; // 用 户 名 

$password = "111"; /密码 

S$dbName = "db_database20"; /| 数据 库 名 

$connlD = mysql_connect($host, $userName, $password); /| 连接 MySQL 数据 库 
mysql_select_db($dbName, $connID); /| 选择 数据 库 
mysql_close($connID); /| 断 开 与 数据 库 连 接 
mysql_select_db($dbName, $connlD); // 再 次 选择 数据 库 


?> 


运行 本 实例 ， 将 在 页 面 中 输出 如 图 20.10 所 示 的 错误 信息 。 从 TRACE 


EE bus/focahostt? |1+ x | P Bas= 


错误 信息 中 可 以 判断 ,第 一 次 对 数据 库 选 择 操作 是 成 功 的 , 而 第 二 
次 操作 是 失败 的 ， 即 可 证 明 mysql_close0 函 数 成 功 关闭 了 与 数据 库 
的 连接 。 


ET EECTTT 本 


图 20.10 错误 提示 
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20.3 管理 MySQL 数据 库 中 的 数据 


PHP 操作 MySQL 数据 库 是 PHP 中 基础 且 较 重要 的 技术 ， 开 发 人 员 只 有 熟练 地 掌握 这 部 分 知识 才能 够 
独立 开发 出 基于 PHP 的 数据 库 应 用 。 

本 节 通 过 一 个 公告 栏 进一步 讲解 PHP 通过 使 用 MySQL 操作 函数 实现 对 MySQL 数据 库 进行 增 、 删 改 、 
查 操 作 的 具体 实现 过 程 。 


20.3.1 应 用 insert 命令 动态 添加 公告 信息 


名 g 视频 讲解 : 光盘 \TM\Video\ 第 20 章 \ 应 用 insert 命令 动态 添加 公告 信息 .exe 
在 实现 动态 添加 公告 信息 前 ， 首 先 要 建立 数据 库 及 数据 表 结 构 。 创 建 数 据 表 的 方法 有 两 种 ， 一 种 是 在 
命令 提示 符 下 创建 ， 另 一 种 是 使 用 phpMyAdmin 图 形 化 管理 工具 创建 。 下 面 介绍 这 两 种 创建 数据 库 的 具体 
步骤 和 实现 过 程 。 
在 命令 提示 符 下 创建 数据 表 结 构 
选择 “开始 ”/“ 和 运行” 命令， 打开“ 运行 ”窗口 。 在 “运行 ”窗口 中 输入 “cmd” 命 令 将 弹出 命令 提 
示 符 窗口 。 在 命令 提示 符 下 使 用 如 下 命令 登录 MySQL 数据 库 服务 器 : 
mysql -uroot -p111 
成 功 登录 MySQL 数据 库 服务 器 后 ， 使 用 use 命令 选择 数据 库 db_database20， 命 令 如 下 : 
use db_database20 
选择 数据 库 db_database20 后 ， 输 入 如 下 命令 创建 数据 表 tb_affiche: 
CREATE TABLE 'tb_affiche' ( 
id'INT(4 ) NOT NULL AUTO_INCREMENT PRIMARY KEY ， 
‘title’ VARCHAR( 200 ) CHARACTER SET gb2312 COLLATE gb2312_chinese_ci NOT NULL ， 
"content TEXT CHARACTER SET gb2312 COLLATE gb2312_chinese_ci NOT NULL ， 
createtime' DATETIME NOT NULL 
) ENGINE = MYISAM ; 
利用 phpMyAdmin 图 形 化 管理 工具 创建 数据 表 结 构 
在 phpMyAdmin 图 形 化 管理 工具 中 ， 选 择 数据 库 db_database20， 然 后 按 图 20.11 所 示 的 表 结 构 创 建 
tb_affiche 表 。 
园 服务 器 :localhost ， 忆 者 基诺 : db_database20 四 表 :tb_afFiche 
加 3。 认 要 构 ,Psa 万 失 索 开拓 和 图 S 出 四 mper 从 要 作 二 和 实 到 8 


字 和 组 类 型 村 各 民 性 mn 邓 认 往外 所 作 
[和 in 下 aui_icremedt 国内 X 图 加 开 
= ee 2 hnese dd 殖 EE 
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图 20.11 公告 信息 数据 表 tb_affiche 的 字段 说 明 
下 面 按照 PHP 访问 MySQL 数据 库 的 一 般 步 骤 讲 解 使 用 PHP 中 的 MySQL 操作 函数 向 数据 库 表 添加 数 
据 的 方法 。 
例 20.9 应 用 insert 命令 动态 地 向 数据 库 中 添加 公告 信息 , 然后 应 用 mysql_query0 函 数 将 SQL 语句 发 送 
到 MySQL 服务 器 ， 从 而 完成 将 数据 动态 添加 到 数据 库 的 操作 。 (实例 位 置 : 光盘 \TMNInstance\20\20.9) 
程序 开发 步骤 如 下 。 
(1) 在 左 侧 导航 区 的 添加 公告 信息 的 图 片上 添加 热 区 。 其 代码 如 下 : 


@ 
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<img src="images/image_09.gif" width="202" height="310" border="0" usemap="#Map"> 
<map name="Map"> 
<area shape="rect" coords="30,45,112,63" href=" add_affiche.php"> 
</map> 
(2) 创建 add_affiche.php 页 面 ， 在 该 页 面 中 建立 公告 信息 录入 表单 ， 并 指定 表单 的 提交 地 址 为 check_ 
add_affiche php 页 面 。 其 代码 如 下 : 
<form name="form1" method="post" action="check_add_affiche.php"> 
<table width="520" height="212" border="0" cellpadding="0" cellspacing="0" bgcolor="#FFFFFF"> 
<tr> 
<td width="87" align="center"> 公 告 主题 : </td> 
<td width="433" height="31"> 
<input name="txt_titile" type="text" id="txt_titile" size="40"> 
*</td> 
</tr> 
<tr> 
<td height="124" align="center"> 公 告 内 容 : </td> 
<td><textarea name="txt_content" cols="50" rows="8" id="txt_content"></textarea></td> 
</tr> 
<tr> 
<td height="40" colspan="2" align="center"> 
<input name="Submit” type="submit" class="btn_grey"” value=" 保存" onClick="return 
check(form1)"> 
<input type="reset" name="Submit2" value=" 重 置 "> 
</td> 
</tr> 
</table> 
</form> 
提交 表单 前 ， 要 对 用 户 输入 数据 的 合法 性 进行 验证 ， 以 便 提高 系统 的 安全 性 。 公 告 录 入 表单 中 所 要 录 
入 的 数据 只 需 验 证 是 否 为 空 即 可 。 该 部 分 是 采用 JavaScript 脚本 在 客户 端 进行 验证 的 ， 这 可 以 有 效 降低 服务 
器 的 负载 。 实 现代 码 如 下 : 
<script language="javascript"> 


function check(form){ // 验 证 表单 信息 是 否 为 空 
if(form.txt_title.value==""}{ // 公 告 标题 信息 不 能 为 空 ， 否 则 弹出 提示 信息 
alert(" 请 输入 公告 标题 "");form .txt_title.focus();return false; 
ifform .txt_content.value==""){ /公告 内 容 不 能 为 空 ， 否 则 弹出 提示 信息 
alert(" 请 输入 公告 内 容 ");form.txt_content.focus();return false; 
} 
form.submit(); /提交 表单 信息 到 数据 处 理 页 
} 
</script> 


(3) 提交 表单 信息 到 数据 处 理 页 check_add_affiche php， 连 接 MySQL 数据 库 服 务 器 ， 并 指定 数据 库 的 
编码 格式 为 GB2312。 通 过 POST 方法 获取 表单 传递 过 来 的 值 ， 然 后 应 用 insert 命令 将 表单 信息 添加 到 数据 
表 ， 并 由 mysql_ query0O 函 数 发 送 到 服务 器 ， 从 而 完成 公告 信息 的 添加 。 添 加 成 功 则 弹出 提示 信息 ， 并 重新 
定位 到 首页 index.php。 其 代码 如 下 : 

<?php 
$conn=mysql_connect("localhost","root","111") or die(" 数 据 库 服务 器 连接 错误 ".mysql_error()); 
mysql_select_db("db_database20",$conn) or die(" 数 据 库 访问 错误 ".mysql_error()); 
Areerossewsts 这 里 必须 指定 数据 库 的 编码 方式 为 GB2312， 否 则 存储 到 数据 表 中 的 公告 信息 为 乱码 
mysql_query("set names gb2312"); /| 选择 编码 格式 为 GB2312 
Stitle=$_POST[txt_title]; // 获 取 公 告 标题 信息 
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$content=$_POST[txt_content]; // 获 取 公 告 内 容 
Screatetime=date("Y-m-d H:i:s"); // 获 取 系 统 的 当前 时 间 ，“H” 表 示 24 小 时 制 必须 大 写 
ren 应 用 mysql_query() 函 数 执行 insert…into 语句 发 送 到 | 


$sql=mysql_query("insert into tb_affiche(title,content,createtime)values('$title','$content','$createtime")"); 
echo "<script>alert(' 公 告 信息 添加 成 功 !");window.location.href= add_affiche.php';</script>"; 


mysql_free_result($sql); /关闭 记录 集 
mysql_close($conn); /| 关闭 MySQL 数据 库 服 务 器 
?> 


在 上 面 的 代码 中 ，date0 函 数 用 来 获取 系统 的 当前 时 间 ， 在 公告 信息 添加 成 功 后 ， 应 用 JavaScript 脚本 
弹出 提示 对 话 框 ， 并 在 JavaScript 脚本 中 应 用 “window.location.href='index.php” 重 定位 网 页 ， 这 里 将 网 页 重 
新 定位 到 首页 。 

(4) 在 正 浏 览 器 中 输入 地 址 ， 按 Enter 键 ， 单 击 “ 添 加 公告 信息 ” 超 链 接 ， 在 页 面 中 添加 公告 主题 和 
公告 内 容 ， 单 击 “ 保 存 ” 按 钮 ， 弹 出 “公告 信息 添加 成 功 ” 提 示人 信息， 运行 结果 如 图 20.12 所 示 。 单 击 “ 确 
定 ”按钮 ， 重 新 定位 到 首页 。 


20.3.2 ”应 用 select 命令 查询 公告 信息 


儿 视频 讲解 : 光盘 \TM\Video\ 第 20 章 \ 应 用 select 命令 查询 公告 信息 .exe 
实现 添加 公告 信息 后 ， 即 可 对 公告 信息 执行 查询 操作 。 
例 20.10 ”应 用 select 命令 动态 检索 数据 库 中 的 公告 信息 ， 使 用 mysql_query0 函 数 将 SQL 语句 发 送 到 
MySQL 服务 器 ， 从 而 完成 数据 的 检索 操作 。《〈 实 例 位 置 : 光盘 \TMNInstance\20\20.10) 
程序 开发 步骤 如 下 。 
(1) 建立 导航 页 面 menu.php， 然 后 应 用 include 语句 将 菜单 导航 页 面 menu.php 嵌入 index.php 页 面 中 。 
(2) 建立 search_affiche.php 页 面 ， 在 search_affiche.php 页 的 右 侧 信息 显示 区 域 添加 一 个 表单 、 一 个 文 
本 框 、 一 个 提交 按钮 。 其 代码 如 下 : 
<form name="form1" method="post" action=""> 
查询 关键 字 &nbsp; 
<input name="txt_keyword" type="text" id="txt_keyword" size="40"> 
<input type="submit" name="Submit" value=" 搜 索 " onClick="return check(form)"> 
</form> 


为 防止 用 户 搜 索 空 信息 , 本 程序 在 保存 按钮 的 onclick 事件 中 , 调用 一 个 由 JavaScript 脚本 定义 的 checkO 
函数 ， 用 来 限制 文本 框 信息 不 能 为 空 。 当 用 户 单 击 “ 保 存 ” 按 钮 时 ， 自 动 调用 check0 函 数 。 该 函数 的 代码 


如 下 : 
<script language="javascript"> 
function check(form){ /验证 表单 信息 是 否 为 空 
ifform _txt_keyword.value=="){ /查询 关键 字 不 能 为 空 ， 否 则 弹出 提示 信息 
alert(" 请 输入 查询 关键 字 "");form.txt_keyword.focus();return false; 
5 
form.submit(); /提交 表单 信息 
并 
</script> 


(3) 提交 表单 信息 到 数据 处 理 页 check_add_affiche.php， 连 接 MySQL 数据 库 服务 器 ， 并 指定 数据 库 的 
编码 格式 为 GB2312。 通 过 POST 方法 获取 表单 传递 过 来 的 值 ， 然 后 应 用 insert 命令 将 表单 信息 添加 到 数据 
表 ， 从 而 完成 公告 信息 的 添加 。 添 加 成 功 则 弹出 提示 信息 ， 并 重新 定位 到 首页 index.php。 其 代码 如 下 : 

<?php 
$conn=mysql_connect("localhost","root","111") or die(" 数 据 库 服 务 器 连接 错误 ".mysql_error()); 
mysql_select_db("db_database20",$conn) or die(" 数 据 库 访问 错误 "mysql_error()); 
mysql_query("set names gb2312"); /选择 编码 格式 为 GB2312 
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$keyword=$_POSTI[txt_keyword]:; // 获 取 查 询 关 键 字 内 容 
$sql=mysql_query("select * from tb_affiche where title like '"%$keyword%' or content like '"%$keyword%"); 
Srow=mysql_fetch_object($sql); // 获 取 查 询 结 果 集 
if(I$rowX{ // 如 果 未 检索 到 信息 资源 ， 则 弹出 提示 信息 
echo "<font color='red'> 您 搜索 的 信息 不 存在 ， 请 使 用 类 似 的 关键 字 进行 检索 !</font>"; 
} 
dof /应 用 do…while 循环 语句 输出 查询 结果 
?> 

<tr bgcolor="#FFFFFF"> 


<td><?php echo $row->title;?></td> 
<td><?php echo $row->content;?></td> 


</tr> 

<?php 

}while($row=mysql_fetch_object($sql)); /do…while 循环 语句 结束 
mysql_free_result($sql); /关闭 记录 集 
mysql_close($conn); /| 关闭 MySQL 数据 库 服 务 器 
?> 


(4) 在 正 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 单 击 “ 查 询 公 告 信息 ” 超 链接 ， 将 弹出 如 图 20.13 所 示 的 
查询 公告 信息 页 面 ， 在 页 面 的 文本 框 中 输入 查询 关键 字 ， 然 后 单 击 “ 搜 索 ” 按 钮 即 可 查询 到 所 有 与 查询 关 
键 字 匹 配 的 公告 信息 。 
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20.12 添加 公告 页 面 的 运行 结果 图 20.13 查询 公告 页 面 的 运行 结果 
20.3.3 ”解决 截取 公告 主题 乱码 问题 


区 4 视频 讲解 : 光盘 \TMIVVideo\ 第 20 章 \ 解 决 截取 公告 主题 乱码 问题 .exe 

在 开发 Web 程序 时 ， 为 了 保持 整个 页 面 的 合理 布局 ， 经 常 需要 对 一 些 较 长 的 主题 进行 截取 输出 ， 但 由 
于 汉字 占有 两 个 字符 ， 如 果 截 取 位 置 选取 不 当 ， 将 会 导致 截取 的 字符 串 末 尾 出 现 乱 码 。 例 如 ， 应 用 substr0 
函数 对 普通 字符 进行 截取 操作 会 非常 方便 ， 但 对 全 角 字 符 进 行 截取 可 能 会 出 现 乱 码 。 可 以 通过 自 定义 函数 
chinesesubstr() 解 决 上 述 问 题 。 解 决 截取 字符 串 乱 码 前 后 的 对 比如 图 20.14 和 图 20.15 所 示 。 


了 [出 现 乱码 

ee 
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ee 


图 20.14 应 用 substr0 函 数 截 取 字符 串 出 现 乱码 20.15 ”应 用 自 定义 函数 chinesesubstr0 解 决 截取 字符 串 乱 码 
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例 20.11 应 用 自 定义 函数 chinesesubstr0 解 决 截取 公告 主题 出 现 乱 码 的 问题 。 (实例 位 置 ， 光盘 \TM\ 
Instance\20\20.11) 
程序 开发 步骤 如 下 。 
(1) 在 菜单 导航 页 menuphp 中 插入 一 张 图 片 image 09.gif， 在 图 片上 添加 热 区 ， 链 接 文件 为 
interceptphp。 然 后 应 用 include 语句 将 菜单 导航 页 menu.php 嵌入 index.php 页 面 中 。 
(2) 建立 interceptphp 页 ， 在 interceptphp 页 的 右 侧 信息 显示 区 域 动态 输出 公告 主题 信息 ， 并 截取 公 
告 主题 的 前 30 个 字符 。 其 代码 如 下 : 
<?php 
$conn=mysql_connect("localhost","root","111") or die(" 数 据 库 服 务 器 连接 错误 ".mysql_error()); 
mysql_select_db("db_database20",$conn) or die(" 数 据 库 访 问 错误 ".mysql_error()); 


mysql_query("set names gb2312"); /| 选择 编码 格式 为 GB2312 
$sql=mysql_query("select * from tb_affiche order by createtime desc limit 0,6"); 
$info=mysql_fetch_array($sql); // 获 取 查 询 结果 集 
if$info==false){ // 如 果 未 检索 到 信息 资源 ， 则 弹出 提示 信息 
echo "本 站 暂 无 公告 信息 "; 
}else{ 
do{ 
?> 
<tr bgcolor="#E3E3E3"> 
<td height="24" align="left" bgcolor="#FFFFFF"><img src="images/xing.gif" width="9" height= 
<?php 
// 调 用 自 定义 函数 chinesesubstr()， 屏 蔽 乱码 
echo chinesesubstr($infoftitle],0,30); 
if(strlen($info[title])>30){ // 如 果 公告 主题 的 长 度 大 于 30 个 字符 ， 则 输出 “…” 
echo ".…"; 
?> 
</td> 
</tr> 
<?php 
}while($info=mysql_fetch_array($sql)); /ldo…while 循环 语句 结束 
} 
mysql_free_result($sql); /关闭 记录 集 
mysql_close($conn); /关闭 MySQL 数据 库 服 务 器 


?> 

在 上 面 的 代码 中 ， 应 用 了 自 定义 函数 chinesesubstr0， 为 了 管理 和 维护 方便 ， 这 里 将 该 函数 封装 在 一 个 
独立 的 fonction.php 页 中 。 因 此 ， 调 用 时 需要 应 用 include 语句 进行 引用 。 其 代码 如 下 : 

<?php include("function.php"); ?> 

其 中 ，function.php 页 中 定义 一 个 用 于 截取 一 段 字 符 串 的 chinesesubstr0 函 数 。 其 代码 如 下 : 


<?php 
function chinesesubstr($str,$start,$len) { //$str 指 字符 串 ，$start 指 字符 串 的 起 始 位 置 ，$len 指 字符 串 长 度 
Sstren=$start+ $len; /用 $strlen 存储 字符 串 的 总 长 度 , 即 从 字符 串 的 起 始 位 置 到 末尾 的 总 长 度 
for($i=0;$i<$strlen;$i++){ 
if(ord(substr($str,$i,1))>0xa0) { // 如 果 字符 串 中 首 个 字 节 的 ASCII 序数 值 大 于 0xa0， 则 表示 汉字 
S$tmpstr.=substr($str,$i,2); /每 次 取出 两 位 字符 赋 给 变量 $tmpstr， 即 等 于 一 个 汉字 
$i++; /变量 自 加 1 
5 
else 
$tmpstr.=substr($str,$i,1); /如 果 不 是 汉字 ， 则 每 次 取出 一 位 字符 赋 给 变量 $tmpstr 
二 
return $tmpstr; /返回 字符 串 
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} 


?> 

(3) 在 正 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 系 统 自动 按 添 加 公告 的 时 间 降 序 排列 ， 应 用 limit 语句 取 
出 最 新 的 6 条 公告 主题 信息 ， 每 条 公告 主题 仅 截取 前 30 个 字符 进行 输出 ， 公 告 主题 大 于 30 个 字符 的 字符 
串 部 分 用 省 略 号 (… ) 代替， 运行 结果 如 图 20.16 所 示 。 如 果 未 检索 到 相 匹配 的 公告 信息 ， 则 输出 提示 信息 。 


和 所 长 在 线 考证 网 > > | 


图 20.16 解决 截取 公告 主题 乱码 问题 
20.3.4 分 页 显示 公告 信息 


名 视频 讲解 :光盘 \TM\Video\ 第 20 章 \ 分 页 显示 公告 信息 .exe 
在 实现 了 添加 公告 信息 后 ， 便 可 以 对 公告 信息 执行 查询 操作 。 下 面 在 实例 20.9 的 基础 上 实现 查询 公告 信息 。 
例 20.12 ”应 用 select 命令 动态 检索 数据 库 中 的 公告 信息 ， 并 应 用 mysql_query0 函 数 将 SQL 语句 发 送 
到 MySQL 服务 器 ， 从 而 完成 数据 的 检索 操作 。〔 实 例 位 置 ， 光盘 \TMNInstance\20\20.12) 
程序 开发 步骤 如 下 。 
(1) 将 菜单 导航 部 分 封装 成 一 个 独立 的 页 menu.php， 在 该 页 中 插入 一 张 图 片 image_09.gif， 在 图 片上 
添加 热 区 ， 链 接 page_affiche.php 文件 。 
(2) 在 page_affiche.php 页 的 右 侧 信息 显示 区 域 ， 连 接 MySQL 数据 库 服务 器 ， 连 接 数 据 库 ， 并 指定 数 
据 库 的 编码 格式 为 GB2312 编码 。 应 用 select 命令 将 检索 公告 信息 表 中 的 数据 ， 并 由 mysql_query0 函 数 发 送 
到 服务 器 ， 从 而 完成 公告 信息 的 分 页 显示 。 其 代码 如 下 : 
<table width="550" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" bgcolor="#999999"> 
<tr align="center bgcolor="#fOfOf0"> 
<td width="221"> 公 告 标题 </td> 
<td width="329"> 公 告 内 容 </td> 
</tr> 
<?php 
$conn=mysql_connect("localhost","root","111") or die(" 数 据 库 服 务 器 连接 错误 ".mysql_error()); 
mysql_select_db("db_database20",$conn) or die(" 数 据 库 访问 错误 ".mysql_error()); 
这 里 必须 指定 数据 库 的 编码 方式 为 GB2312， 否 则 输出 到 浏览 器 中 的 公告 信息 为 乱码 **** ******x**/ 
mysql_query("set names gb2312"); 
tt page 为 当前 页 ， 如 果 $page 为 空 ， 则 初始 化 为 1 


if (Spage=="™"X{ 
$page=1;} 
if (is_numeric($page)){ // 判 断 变 量 $page 是 否 为 数字 , 如 果 是 则 返回 true 
Spage_size=4; /每 页 显示 4 条 记录 
$query="select count(*) as total from tb_affiche order by id desc"; 
$result=mysql_query($query); /查询 符合 条 件 的 记录 总 条 数 


@ 


第 20 章 数据 库 编程 技术 


$message_count=mysql_result($result,0,"total"); // 要 显示 的 总 记录 数 
os 由 记录 总 数 除 以 每 页 显示 的 记录 数 求 出 所 分 的 页 著 季 re 
$page_count=ceil($message_count/$page_size); 
S$offset=($page-1)*$page_size; // 计 算 下 一 页 从 第 几 条 数据 开始 循环 
$sql=mysql_query("select * from tb_affiche order by id desc limit $offset, $page_size"); 
Srow=mysql_fetch_object($sql); // 获 取 查 询 结果 集 
if(!$row)}{ // 如 果 未 检索 到 信息 资源 ， 则 输出 提示 信息 
echo "<font color='red'> 暂 无 公告 信息 !</font>"; 
’ 
do{ 
2 
<tr bgcolor="#FFFFFF"> 
<td><?php echo $row->title;?></td> 
<td><?php echo $row->content;?></td> 
</tr> 
<?php 
}while($row=mysql_fetch_object($sql)); 
国 
</table> 


人 注意 do…while0 循 环 和 while0 循 环 的 区 别 : do*…while0 循 环 是 先 执行 {} 中 的 代码 段 ， 然 后 判断 while 中 

的 条 件 表达 式 是 否 成 立 ， 如果 返 回 true, 则 重复 输出 们 中 的 内 容 , 否则 结束 循环 , 执行 while 下 面 的 语句 ; 
while 循环 是 先 判 断 while 中 的 表达 式 ， 当 返回 true 时 ， 再 执行 “{}” 中 的 代码 。 两 者 的 主要 区 别 在 于 ， 
do*…while() 循 环比 while0 循 环 多 输出 一 次 结果 。 


(3) 添加 一 个 一 行 一 列 的 表格 ， 添 加 如 下 代码 ， 实 现 翻 页 功能 。 
<table width="550" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<!-- 翻 页 条 --> 
<td width="37%">&nbsp;&nbsp; 页 次 : <?php echo $page;?>/<?php echo $page_count;?> 页 &nbsp; 记 
录 : <?php echo $message_count;?> 条 &nbsp; </td> 
<td width="63%" align="right"> 
<?php 
上 如 果 当 前 页 不 是 首页 */ 
if($page!=1){ 
六 显示 “首页 ” 超 链接 */ 
echo "<a href=page_affiche.php?page=1> 首 页 </a>&nbsp;"; 
/* 显示 “上 一 页 ” 超 链 接 */ 
echo "<a href=page_affiche.php?page=".($page-1)." > 上 一 页 </a>&nbsp;"; 


} 

记 ” 如 果 当前 页 不 是 尾 页 */ 

if($page<$page_count}{ 

请 显示 “下 一 页 ” 超 链接 */ 

echo "<a href=page_affiche.php?page=".($page+1)."> 下 一 页 </a>&nbsp;"; 
六 显示 “ 尾 页 ” 超 链接 */ 

echo "<a href=page_affiche.php?page=".$page_count."> 尾 页 </a>"; 


让 
mysql_free_result($sql); /关闭 记录 集 
mysql_close($conn); /| 关闭 MySQL 数据 库 服务 器 
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</tr> 


</table> 


(4) 在 正 浏 览 器 中 输入 地 址 ， 按 Enter 键 ， 分 页 显 


20.3.5 ”应 用 update 命令 动态 编辑 公告 信息 


示 公 告 信息 ， 运 行 结果 如 图 20.17 所 示 。 


Nn 


视频 讲解 : 光盘 \TM\Video\ 第 20 章 \ 应 用 update 命令 动态 编辑 公告 信息 .exe 

公告 信息 也 不 是 - -成 不 变 的 ， 可 以 对 公告 主题 及 公告 内 容 的 动态 信息 进行 编辑 。 

例 20.13 应 用 update 命令 动态 编辑 数据 库 中 的 公告 信息 ， 然 后 通过 mysql_query0 函 数 将 SQL 语句 发 
送 到 MySQL 服务 器 ， 从 而 完成 数据 的 编辑 操作 。 (实例 位 置 : 光盘 \TM\Instance\20\20.13) 

程序 开发 步骤 如 下 。 

(1) 在 菜单 导航 页 menu.php 的 image_09.gif 图 片上 添加 热 区 ， 链 接 update_affiche.php 文件 。 


有 意 为 了 提高 效率 ， 将 首页 index.php 另存 为 update_affiche.php。 


(2) 在 update_affiche.php 页 的 右 侧 信息 显示 区 域 应 用 select 命令 检索 出 全 部 的 公告 信息 ， 添 加 链接 文 
件 modify.php， 并 向 modify.php 页 传递 公告 ID 的 参数 值 。 其 代码 如 下 : 
<a href="modify.php?id=<?php echo $row->id;?>"> 
<img src="images/update.gif" width="20" height="18" border="0"> 
</a> 
(3) 按照 首页 的 风格 创建 一 个 编辑 公告 信息 的 modify.php 页 , 在 该 页 面 中 添加 一 个 表单 、 一 个 文本 框 、 
-个 编辑 框 、 一 个 隐藏 域 、 一 个 提交 按钮 和 一 个 重 置 按钮 , 设置 表单 的 action 属性 值 为 check_modify_ok.php。 
其 代码 如 下 : 
<?php 
$conn=mysql_connect("localhost","root","111") or die(" 数 据 库 服务 器 连接 错误 ".mysql_error()); 
mysql_select_db("db_database20",$conn) or die(" 数 据 库 访问 错误 ".mysql_error()); 
mysql_query("set names gb2312"); 


$id=$_GET[id]; // 应 用 GET 方法 接收 欲 编辑 的 公告 ID 
$sql=mysql_query("select * from tb_affiche where id=$id"); /检索 公告 ID 所 对 应 的 公告 信息 
$row=mysql_fetch_object($sql); /获取 结果 集 

?> 


<form name="form1" method="post" action="check_modify_ok.php"> 
<table width="520" height="212" border="0" cellpadding="0" cellspacing="0" bgcolor="#FFFFFF"> 
<tr> 
<td width="87" align="center"> 公 告 主题 : </td> 
<td width="433" height="31"> 
<input name="txt_title" type="text" id="txt_title" size="40" value="<?php echo $row->title;?>"> 


< 将 公告 ID 的 信 赋 师 随 且 庆 于 记 汪 2 3 二 3 
<input name="id" type="hidden" value="<?php echo $row->id;?>"></td> 

</tr> 

<tr> 


<td height="124" align="center"> 公 告 内 容 : </td> 
<td><textarea name="txt_content"> <?php echo $row->content;?></textarea></td> 
</tr> 
<tr> 
<td height="40" colspan="2" align="center > 
<input name="Submit" type="submit" class="btn_grey" value=" 修 改 " onClick="return check(form1);"> 
<input type="reset" name="Submit2" value=" 重 置 "></td> 
</tr> 
</table> 
</form> 
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(4) 提交 表单 信息 到 数据 处 理 页 check_modify_ok.php。 首 先 ， 连 接 MySQL 数据 库 服务 器 ， 然 后 选择 
数据 库 , 设置 GB2312 的 编码 格式 。 应 用 POST 方法 获取 表单 信息 , 接 下 来 应 用 mysql_query0 函 数 向 MySQL 
数据 库 服务 器 发 送 修改 公告 信息 的 SQL 语句 ， 最 后 应 用 让 else 条 件 语 句 对 修改 后 的 信息 进行 判断 ， 并 弹 
出 相应 的 提示 信息 ， 重 新 定位 到 公告 信息 编辑 页 。 其 代码 如 下 : 

<?php 

oo 连接 数据 源 ,读者 可 将 此 处 封装 成 独立 的 页 ,然后 应 用 include 语句 调用 ,效率 会 很 高 ** 
$conn=mysql_connect("localhost","root","111") or die(" 数 据 库 服务 器 连接 错误 ”mysql_error()); 
mysql_select_db("db_database20",$conn) or die(" 数 据 库 访问 错误 ".mysql_error()); 

mysql_query("set names gb2312"); 


TT 
Stitle=$_POST[txt title]; /获取 更 改 的 公告 主题 
$content=$_POST[txt_content]; 1/ 获取 更 改 的 公告 内 容 
$id=$_POSTIid]; /获取 更 改 的 公告 ID 
/应 用 mysql_query() 函 数 向 MySQL 数据 库 服务 器 发 送 修改 公告 信息 的 SQL 语句 
$sql=mysql_query("update tb_affiche set title='$title,content='$content' where id=$id"); 
说 $sql){ 

echo "<script>alert(' 公 告 信息 编辑 成 功 !");history.back();window.location.href='modify.php?id=$id';</script>"; 
Jelse{ 

echo "<script>alert(' 公 告 信息 编辑 失败 !");history.back();window.location.href='modify.php?id=$id';</script>"; 
由 


?> 

(5) 在 下 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 在 页 面 中 输入 查询 关键 字 ， 单 击 “ 搜 索 ” 按 钮 ， 即 可 输出 

检索 到 的 公告 信息 资源 ， 单 击 侈 超 链接 ， 弹 出 编辑 公告 详细 信息 页 ， 对 指定 的 公告 信息 进行 编辑 ， 单 击 “ 修 
改 ” 按 钮 ， 即 可 完成 指定 公告 信息 的 编辑 操作 ， 运 行 结 果 如 图 20.18 所 示 。 
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图 20.17 分 页 显示 公告 信息 的 运行 结果 图 20.18 编辑 公告 页 面 的 运行 结果 


20.3.6 ”应 用 delete 命令 动态 删除 公告 信息 


名 4 视频 讲解 : 光盘 \TM\Video\ 第 20 章 \ 应 用 delete 命令 动态 删除 公告 信息 .exe 

公告 信息 是 对 一 些 重要 的 、 新 鲜 的 事务 进行 管理 ， 因 此 ， 为 了 节省 系统 资源 ， 可 以 定期 对 公告 主题 及 
公告 内 容 的 信息 进行 删除 。 

例 20.14 应 用 delete 命令 动态 编辑 数据 库 中 的 公告 信息 ， 然 后 通过 mysql_query0 函 数 将 SQL 语句 发 
送 到 MySQL 服务器， 从 而 完成 数据 的 删除 操作 。〔 实 例 位 置 光盘 \TMNInstance\20\20.14) 

程序 开发 步骤 如 下 。 

(1) 在 菜单 导航 页 menu.php 的 image 09.gif 图 片上 添加 热 区 ， 链 接 delete_affiche.php 文件 。 

(2) 在 delete affiche php 页 的 右 侧 信息 显示 区 域 应 用 select 命令 检索 出 全 部 的 公告 信息 。 添 加 链接 文 
件 check_del okphp， 并 向 check _ del _ ok.php 页 传递 公告 ID 的 参数 值 。 其 代码 如 下 : 

@ 
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<a href="check_del_ok.php?id=<?php echo $row->id;?>"> 
<img src="images/delete.gif" width="22" height="22" border="0"> 
</a> 
(3) 提交 公告 信息 的 ID 到 数据 处 理 页 check_del ok.php。 首 先 ， 连 接 MySQL 数据 库 服 务 器 ， 然 后 选 
择 数 据 库 , 设置 GB2312 的 编码 格式 。 应 用 GET 方法 获取 和 欲 删 除 的 公告 ID， 接 下 来 应 用 mysql_queryO 函 数 
向 MySQL 数据 库 服 务 器 发 送 删除 公告 信息 的 SQL 语句 , 最 后 应 用 站 …else 条 件 语句 对 修改 后 的 信息 进行 判 
断 ， 并 弹出 相应 的 提示 信息 ， 重 新 定位 到 公告 信息 编辑 页 。 其 代码 如 下 : 
<?php 
$conn=mysql_connect("localhost","root","root") or die(" 数 据 库 服 务 器 连接 错误 ".mysql_error()); 
mysql_select_db("db_database20",$conn) or die(" 数 据 库 访问 错误 ".mysql_error()); 
mysql_query("set names gb2312"); 
$id=$_GET[id]; 
$sql=mysql_query("delete from tb_affiche where id=$id"); 
if($sqlX{ 
echo "<script>alert( 公告 信息 删除 成 功 ! '):history.back();window.location.href='delete_affiche.php?id=$id'; 
</script>"; 
Jelse{ 
echo "<script>alert( 公告 信息 删除 失败 ! '):history.back();window.location.href='delete_affiche.php?id=Sid'; 
</script> 
} 


?> 


[BE 由 于 数据 处 理 页 check_del ok php 中 都 是 动态 代码 ,没有 指定 编码 格式 为 GB2312 编码 类 型 ， 以致 
读者 在 用 Dreamweaver 开发 工具 打开 该 文件 时 ， 中 文部 分 将 会 显示 乱码 。 为 了 解决 这 一 问题 ， 读 者 可 以 


在 该 页 指定 其 编码 格式 。 其 代码 如 下 : 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312"> 


(4) 在 下 浏览 器 中 输入 地 址 ， 按 Enter 键 ， 在 页 面 中 输入 查询 关键 字 ， 单 击 “ 搜 索 ” 按 钮 ， 即 可 输出 
检索 到 的 公告 信息 资源 。 单 击 指定 公告 信息 后 面 的 拿 按 钮 ， 弹 出 删除 公告 信息 提示 ， 然 后 单 击 “ 确 定 ” 按 
钮 ， 即 可 完成 对 指定 公告 信息 的 删除 操作 ， 运 行 结果 如 图 20.19 所 示 。 
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图 20.19 删除 公告 页 面 信息 的 运行 结果 
20.4 PHP 操作 MySQL 事务 
事务 处 理 机 制 在 程序 开发 过 程 中 有 着 非常 重要 的 作用 ， 它 可 以 使 整个 系统 更 加 安全 。 例 如 ， 在 银行 处 
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理 转账 业务 时 ， 如 果 A 账户 中 的 金额 刚 被 发 出 ， 而 B 账户 还 没 来 得 及 接受 就 发 生 停电 ， 会 给 银行 和 个 人 带 
来 很 大 的 经 济 损失 。 采 用 事务 处 理 机 制 ， 一 旦 在 转账 过 程 中 发 生意 外 ， 则 程序 将 回 深 ， 不 做 任何 处 理 。 

下 面 讲 解 使 用 事务 处 理 技术 实现 关联 表 信 息 的 删除 方法 。 

例 20.15 将 采用 事务 处 理 方式 ， 对 学 生 信息 表 和 学 生成 绩 表 中 的 数据 进行 删除 。 运 行 本 实例 ， 学 生 信 
息 表 及 学 生成 绩 表 信息 分 别 如 图 20.20 和 图 20.21 所 示 。 当 删除 图 20.20 中 的 学 生 信息 后 ， 查 看 学 生成 绩 信 
息 时 可 以 发 现 ， 与 该 学 生 对 应 的 成 绩 也 被 全 部 删除 。《〈 实 例 位 置 : 光盘 \TMNInstance\20\20.15) 

程序 开发 步骤 如 下 。 

(1) 建立 数据 库 及 数据 表 并 实现 与 数据 的 连接 。 其 代码 如 下 : 

<?phI 

Se re mysqli("localhost","root","111","db_database20"); 

$conn->query("set names utf8"); 

?> 


(2) 显示 所 有 学 生 的 基本 信息 。 其 代码 如 下 : 
<?php 
include_once("conn.php"); 
$sql=$conn->query("select * from tb_stu"); 
Sinfo=$sql->fetch_array(MYSQLI_ASSOC); 
if($info==NULL) 


对 
echo " 暂 无 学 生 信息 六 ; 


?> 


<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo $info[sname];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo S$info[sno];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo $info[sage];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo $info[saddress];?></div></td> 
<td height="25" bgcolor="#FFFFFF"><div align="center"><?php echo $info[ssfzh]:?></div></td> 
<td bgcolor="#FFFFFF"><div align="center"><a href="javascript:if(window.confirm(' 确 定 删除 该 学 生 信 
息 么 ?')==true){window.location.href='delete.php?id=<?php echo S$info[id];?>";}"> 删 除 </a></div></td> 
</tr> 
<?php 


while($info=$sql->fetch_array(MYSQLI_ASSOC)); 


?> 

在 实现 该 模块 功能 时 ， 可 以 利用 JavaScript 实现 该 页 与 delete.php 页 面 的 信息 传递 。 其 代码 如 下 : 

<a href="javascript:if(window.confirm(' 确 定 删除 该 学 生 信息 旷 ')==true) 

{window.location.href='delete.php?id=<?php echo $info[id];?>":}"> 删 除 

</a> 

window 对 象 的 confirm0 方 法 用 于 弹出 一 个 对 话 框 ， 提 示 用 户 真正 删除 某 学 生 的 所 有 信息 。 
(3) 利用 事务 处 理 机 制 对 学 生 的 基本 信息 进行 删除 。 其 代码 如 下 : 

<?php 

$id=$_GETT[id]; 

include_once("conn.php"); 

$conn->autocommit(false); 

if(I$conn->query("delete from tb_sco where id=".$id."")) 

{ 
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$conn->rollback(); 
} 
if(!$conn->query("delete from tb_stu where id=".$id."™")) 


$conn->rollback(); 


$conn->commit(); 

$conn->autocommit(true); 

echo "<script>window.location.href='index.php';</script>"; 
?> 


学 生成 续 管理 系统 
站 名 时 二 Wr ad EE | 
oom 本 站 ee | 
图 20.20 查看 学 生 信息 图 20.21 查看 学 生成 绩 


20.5 PHP 操作 MySQL 存储 过 程 


MySQL 5.0 以 后 的 版 本 开始 支持 存储 过 程 ， 存 储 过 程 具有 一 致 性 、 高 效 性 、 安 全 性 和 体系 结构 等 特点 ， 
本 节 将 具体 讲解 PHP 是 如 何 操纵 MySQL 存储 过 程 的 。 
下 面 介绍 如 何 使 用 存储 过 程 实现 用 户 登 录 。 
例 20.16 使 用 MySQL 存储 过 程 实现 用 户 登 录 身 份 的 验证 。 运 行 本 实例 ， 如 图 20.22 所 示 。 在 图 中 登 
录 窗 口 的 文本 框 中 输入 用 户 名 和 密码 ， 然 后 单 击 “ 登 录 ” 按 钮 ， 如 果 用 户 登 录 的 用 户 名 和 密码 正确 ， 则 会 
将 页 面 转向 如 图 20.23 所 示 的 登录 成 功 提示 页 面 。 实 例 位置 光盘 \TM\Instance\20\20.16) 
程序 开发 步骤 如 下 。 
(1) 创建 存储 过 程 ， 通 过 query0 方 法 调用 存储 过 程 。 其 代码 如 下 : 
delimiter // 
create procedure pro_login1(in un varchar(50), in pass varchar(50)) 
begin 
Ed * from tb_login where username=un and password=pass; 
end 
I 
(2) 通过 MySQLi 扩展 库 建立 与 MySQL 数据 库 的 连接 ， 并 设置 数据 库 字 符 集 为 utff-8。 其 代码 如 下 : 
<?phl 
Se mysqli("localhost","root","111","db_database20"); 
$conn->query("set names utf8"); 
?> 


(3) 建立 用 户 登 录 表 单 ， 当 用 户 在 表单 中 输入 用 户 名 和 密码 ， 并 单 击 “ 提 交 ” 按 钮 后 ， 通 过 如 下 代码 
验证 用 户 的 登录 信息 是 否 正确 。 

<?php 

ifisset($_POST[username]) && trim($_POST[username'"])!=") 

{ 


require_once 'Db.php'; 

$username = trim($_POST[usemame']); 

S$password = trim($_POST[password]); 

$sql = $mysqli->query("call pro_login1(".$username.", ".$password.™)"); 


@ 
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$info = $sql->fetch_array(MYSQLI_ASSOC): 
if($info != null}{ 
$_SESSION[IoginUsername]= $usermame; 
echo '<script>window .location.href="success.php";</script>"; 
Jelse{ 
echo '<div style="width:300px; height:30px; line-height:30px; border1px solid #E59B04; 
background-color:#FCF2E0; color:#FF0000;"> 用 户 名 或 密码 输入 有 误 </div>'; 
站 
出 
?> 
上 述 代码 中 , 首先 判断 $_POST[msemame'] 的 值 是 否 被 设置 , 则 首先 使 用 require_once 语句 包含 conn.php 
文件 ， 然 后 使 用 MySQLi 扩展 库 的 query0 方 法 执行 存储 过 程 pro_login， 并 使 用 query0 方 法 返回 的 对 象 调用 
fetch_arrayO 获 得 结果 集 ， 最 后 通过 判断 结果 集 是 否 为 null 来 判断 用 户 所 输入 的 信息 是 否 正确 。 


Ce | 


‘A 
i | 吾 5 而 - 兽 中 - 
用 户 登 录 GE 和 EE. 基 宫 ee， 人 已经 成 功 虹 录 本 站! | 
本 全 节 Intreret| 作 生起 站 用 从- 00% ~ 
图 20.22 用 户 登录 窗口 图 20.23 ”登录 成 功 的 提示 信息 


20.6 实 战 


敬 i 视频 讲解 ， 光盘 \TM\Video\ 第 20 章 \ 实 战 .exe 

MySQL 数据 库 函 数 的 使 用 方法 是 每 位 PHP 开发 人 员 所 必须 掌握 的 , 只 有 掌握 了 这 些 函 数 的 使 用 方法 才 
能 具备 最 基本 数据 库 系 统 开发 的 能 力 。 为 了 巩固 本 章 所 讲 的 内 容 ， 下 面 通过 具体 实例 进一步 讲解 和 深化 
MySQL 数据 库 函 数 的 应 用 及 PHP 操纵 MySQL 数据 库 进行 增 、 删 、 改 、 查 的 操作 方式 。 


20.6.1 输入 页 码 跳 转 到 指定 页 


在 网 站 的 应 用 过 程 中 ， 经 常 需要 将 数据 库 中 很 多 记录 显示 到 页 面 中 来 ， 如 果 将 所 有 的 记录 在 一 个 页 面 
中 显示 ， 会 给 浏览 者 带 来 很 多 麻烦 ， 最 好 的 解决 方法 就 是 使 用 分 页 技术 ， 将 记录 进行 分 页 显示 。 通 过 分 页 
读 取 记录 信息 ， 在 每 页 中 显示 几 条 记录 ， 这 样 既 方便 浏览 者 的 浏览 ， 也 节省 了 页 面 的 空间 。 

例 20.17 以 分 页 的 方式 显示 员工 信息 ， 并 通过 输入 页 码 跳 转 到 指定 的 页 。〔 实 例 位 置 ， 光 盘 \TM\ 
Instance\20\20.17) 

程序 开发 步骤 如 下 。 

(1) 首先 建立 与 MySQL 数据 库 的 连接 , 并 选择 数据 库 db_database20, 如 果 发 生 错误 则 使 用 mysql_error() 
函数 输出 错误 号 ,最 后 使 用 mysql_query0 函 数 执行 set names gb2312 命令 来 设置 数据 库 的 字符 集 为 简体 中 文 。 
其 代码 如 下 : 

<?php 

$id=mysql_connect("localhost","root","111")or dir(' 连 接 失 败 :' . mysql_error()); 

if(mysql_select_db("db_database20",$id)) 


@ 
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echo ™; 
else 
echo ("连接 失败 :' . mysql_error()); 
mysql_query("set names gb2312"); 
> 
(2) 使 用 分 页 的 方式 显示 员工 信息 。 其 代码 如 下 : 
<?php 
include("conn/conn.php"); 
if Nt =="") {$page=1;}if ($jl=="") { 


上 
?> 
<body> 
<table width="450" border="1" cellpadding="0" cellspacing="0" bgcolor="#4DB1FD"> 
<tr> 
<td width="100" height="25" align="center" class="STYLE2"> 姓 名 </td> 
<td width="100" align="center" class="STYLE2"> 编 号 </td> 
<td width="125" align="center" class="STYLE2"> 电 话 </td> 
<td width="175" align="center" class="STYLE2"> 地 址 </td> 
</tr> 
<?php 
if($page}{ 
$page_size=2; /每 页 显示 两 条 记录 
$query="select count(*) as total from tb_insert where id"; /从 数据 库 中 读 取 数 据 
$result=mysql_query($query); 
$message_count=mysql_result($result,0,"total"); // 获 取 总 的 记录 数 
$page_count=ceil($message_count/$page_size); // 获 取 总 的 页 数 
S$offset=($page-1)*$page_size; 
$query="select * from tb_insert where id order by id desc limit $offset, $page_size"; 
Sresult=mysql_query($query); 
while ($myrow=@mysql_fetch_array($result)X{ 
?> 
<tr> 
<td height="25" align="center"><span class="STYLE2"><?php echo $myrow[name];?></span></td> 
<td align="center"><span class="STYLE2"><?php echo $myrow[number];?></span></td> 
<td align="center"><span class="STYLE2"><?php echo $myrowl[tel];?></span></td> 
<td align="center"><span class="STYLE2"><?php echo $myrow[address];?></span></td> 
</tr> 
<?php 
} 
} 


?> 
</table> 
<table width="450" border="1" cellpadding="0" cellspacing="0" bgcolor="#4DB1FD"> 
<tr> 
<td width="45%" align="center"><span class="STYLE1">&nbsp;&nbsp; 页 次 : 
<?php echo $page;?>/ <?php echo $page_count;?> 页 记录 : 
<?php echo $message_count;?> 条 &nbsp; </span> 
</td> 
<td width="55%" height="22" align="center"><span class="STYLE1"> 分 页 : 
<?php 
if($page!=1X{ 


echo "<a href=index.php?page=1> 首 页 </a>&nbsp;"; 
@ 
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echo "<a href=index.php?page=".($page-1)." > 上 一 页 </a>&nbsp;"; 


} 

if($page<$page_count}{ 
echo "<a href=index.php?page=".($page+1)."> 下 一 页 </a>&nbsp;"; 
echo "<a href=index.php?page=".$page_count."> 尾 页 </a>"; 


2> } 
</span></td> 
</tr> 

</table> 

运行 上 述 实例 ， 结 果 如 图 20.24 所 示 ， 在 图 中 的 文本 框 中 下 好 
输入 要 浏览 的 页 码 ， 然 后 单 击 “ 跳 转 ” 按 钮 即 可 跳 转 到 要 浏览 。 轩辕 
的 页 面 。 责 Ri17 3 页 记录 8 来。 而 :下 -而 必 而 LU 
20.6.2 ”图 片 的 分 栏 分 页 显示 ee 


PHP 实现 MySQL 数据 库 中 的 数据 分 页 显示 相对 简单 ,在 此 基础 上 还 可 以 实现 如 图 片 、 代 码 块 的 分 栏 分 
页 显示 。 

例 20.18 图 片 的 分 栏 分 页 显示 。〔 实 例 位 置 ， 光盘 \TM\Instance\20\20.18) 

程序 开发 步骤 如 下 : 

(1) 创建 数据 库 连 接 。 其 代码 如 下 : 

<?php 

$id=mysql_connect("localhost","root","111")or die(' 连 接 失 败 :'. mysql_error()); 

if(mysql_select_db("db_database20",$id)) 

echo ™ 

else 

echo (连接 失败 . mysql_error()); 

mysql_query("set names gb2312"); 

?> 


(2) 通过 分 页 、 分 类 、 分 栏 方法 将 数据 库 中 的 数据 显示 到 页 面 中 。 其 代码 如 下 : 
<table width="400" border="0" cellpadding="0" cellspacing="1" bgcolor="#663300"> 
<?php 

if($pageX{ 
S$page_size=8; 
$query="select count(*) as total from tb_picture where sort="1"; 
$result=mysql_query($query); 
$message_count=mysql_result($result,0,"total"); 
S$page_count=ceil($message_count/$page_size); 
S$offset=($page-1)*$page_size; 
$query="select * from tb_picture where sort="1' limit $offset, $page_size"; 
Sresult=mysql_query($query); 
$row=mysql_fetch_array( $result); 
if($row[sort]==1){ 
for($i=1; $i<=2; $i++) { 
echo "<tr>"; 
if($i==1) { 
$query="select * from tb_picture where sort="1' limit $offset, $page_size"; 
Sresult=mysql_query($query); } 
$j=1; 
while($myrow=mysql_fetch_array($resutlt)){f 
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$id=$myrowfid]; 
if ($j<=4X 
?> 
<td height="50" align="center" valign="top" bgcolor="#FFFFFF" class="STYLE1"> 
<table width="50" border="0" cellspacing="0"> 
<tr align="center" valign="middle"> 
<td width="50" height="50" align="center”valign="middle” background="images/beijing.jpg"> 
<?php echo $myrow[lid];?><img src="image_1.php?recid=<?php echo $myrowlid];?>"></td> 
</tr> 
</table></td> 
<?php 
} 
++9j; 
if(Sj==5X 
break; 


> 
<?php 
for($i=1, $i<=2; $i++) { 
echo "<tr>"; 
人 
<?php 
if($i==1X{ 
$query="select * from tb_picture where sort='2' limit $offset, $page_size"; 
Sresult=mysql_query($query); } 
$j=1; 
while($myrow=mysql_fetch_array($result))f 
$id=$myrow[id]; 
if($j<=4X 
?> 
<td height="50" align="center” valign="top" bgcolor="#FFFFFF" class="STYLE1"> <table 
width="50" border="0" cellspacing="0"> 
<tr align="center" valign="middle"> 
<td width="50" height="50" align="center" valign="middle"” background="images/beijing.jpg"> 
<?php echo $myrow[lid];?><img src="image_1.php?recid=<?php echo $myrow[lid];?>"></td> 


</tr> 
</table> 
</td> 
<?php 
上 
++9j; 
if ($j==5X break; 
} 
有 
echo "</tr>"; 
} 
?> 
</table> 


<table width="400" border="0" cellspacing="0" cellpadding="0"> 
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<tr> 
<td width="50%" height="22" class="hongse01 STYLE1">&nbsp;&nbsp; 页 次 : <font class="huise01"> 
<?php echo $page;?></font> 
/ <font class="huise01"><?php echo $page_count:?> </font> 页 记录 : <font class="huise01"> 
<?php echo $message_count;?> 
</font> 条 &nbsp; </td> 
<td width="39%" height="22" class="hongse01 STYLE1"> 分 页 : 
<?php 
$xsoudh="id=S$id"; 
Snext=$|jl*10; 
$n=S$Ijl-1; 
$m=$ljjl+1; 
$prev_page=$page-10; 
if (Sljl==0X{ 
echo "<img src=\"images/02.jpg\" width=\"8\" height=\"9\" title=\" 首 页 \">"; 
jelsef 
echo "<a href='$PATH_INFO?page=1'><img src=\"images/02.jpg\" width=\"8\" height=\"9\" 
border=\"0\" title=\" 首 页 "></a>&nbsp;"; 
$ccc=$vv-10; 
echo "<a ”href='$PATH_INFO?page=$prev_page'><img src=\"images/01.jpg\" width=\"8\" 
height=\"9v" title=\" 上 十 页 ></a>"; 
} 
?> 
<?php 
for($j=1:9j<=$page_countSj++)f 
$pnext=$next+9j; 
if($mm==10){ break; } 
if($mm>$page_count)}{ break; } 
if($page_count-$vv<10X{ 
if ($mm>=$page_count-$vv){ break; } 


echo "<a href='$PATH_INFO?page=$pnext> $pnext </a>"; 
$mm=$mm+1; 


?> 
</td> 
<td width="11%"> <span class="STYLE1"> 
<?php 
$vwv=$vv+$mm' 
if($page_count-$vv<=0X{ 
echo "<img src=\"images/03.jpg\" width=\"8\" height=\"9\" title=\" 尾 页 ">"; 
Jelse{ 
echo "<a href='$PATH_INFO?page=$pnext><img src=\"images/03.jpg\" width=\"8\" 
height=\"9v" title=\" 下 十 页 ></a>"; 


上 
if($message_count==0){ echo "没有 记录 "; } 
} 


?> 
</span></td></tr> 
</table> 


(3) 读 取 数据 库 中 二 进 制 文件 调用 的 image_1.php 文件 。 其 代码 如 下 : 
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<?php 

S$id=mysql_connect('localhost','root,"111"); 

mysql_select_db('db_database20',$id); 

$query="select * from tb_picture where id=".$recid; 

Sresult=mysql_query($query); 

if(!I$result) die("error: mysql query"); 
$num=mysql_num_rows(S$result); 

if($num<1) die("error: no this recorder"); 
$data = mysql_result($result,0,"xq"); 

mysql_close($id); 

echo $data; 

?> 


运行 本 实例 ， 如 图 20.25 所 示 ， 通 过 页 面 中 的 分 页 链接 即 可 以 分 栏 分 页 的 方式 浏览 所 有 图 片 。 
20.6.3 ”查询 图 书信 息 表 中 的 前 3 条 记录 


例 20.19 查询 图 书信 息 表 tb_book 中 的 前 三 条 图 书信 息 记录 ， 单 击 “ 查 询 ” 按 钮 即 可 查询 前 三 条 图 书 
信息 记录 ， 运 行 结果 如 图 20.26 所 示 。 (实例 位 置 : 光盘 \TM\Instance\20\20.19) 

具体 实现 步骤 如 下 。 

(1) 创建 数据 库 连接 文件 conn.php。 其 代码 如 下 : 


<?php 

$conn=mysql_connect("localhost","root","111"); /| 连接 数据 库 服务 器 
mysql_select_db("db_database20,$conn); // 连 接 db_database20 数据 库 
mysql_query("set names utf8"); // 设 置 数据 库 编码 格式 

i 


(2) 创建 index.php 文件 ， 设 计 图 书信 息 查 询 表单 页 面 ， 效 果 如 图 20.26 所 示 。 
(3) 完成 查询 操作 。 其 代码 如 下 : 


<?php 
include("conn.php"); // 包 含 conn.php 文件 
ifisset($_POST[Submit]) and $_POST[Submit]==" 查 询 "X / 淹 断 用 户 是 否 执行 查询 操作 
$select=mysql_query("select * from tb_book1 where id limit 3 ",$conn); // 查 询 数据 库 中 前 三 条 记录 
}else{ 
S$select=mysql_query("select * from tb_book1",$conn); /查询 所 有 记录 
2 
<?php 
while($rows=mysql_fetch_array($select)){ /使 用 while 循环 查询 结果 
2> 
<tr> 
<td><?php echo $rows['id"];?>&nbsp;</td> <!-- 输 出 id--> 
<td><?php echo $rows[bookname'];?>&nbsp;</td> <!-- 输 出 书 名 --> 
<td><?php echo $rows['maker"];?>&nbsp;</td> <!- 输 出 作者 --> 
<td><?php echo $rows['publisher];?>&nbsp;</td> <!- 输 出 出 版 社 --> 
<td><?php echo $rows[issuDate];?>&nbsp;</td> <!- 输 出 出 版 时 间 --> 
<td><?php echo $rows[price'];?>&nbsp;</td> <!-- 输 出 价格 --> 
</tr> 
<?php 
} /| 结 束 循环 
?> 
运行 结果 如 图 20.26 所 示 。 


@_ 


1 加 2 和 3 3 加 Ca 
5 加 Ba 7 人 sD 
13 1 15 16m 
17 回 18 19» 20 
页 次 :1 /2 页 记录 :12 条 分 页 : M12 Mm 
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图 20.25 图 片 的 分 栏 分 页 显示 
20.6.4 ”对 查询 结果 进行 降序 排列 输出 


例 20.20 ”对 图 书信 息 表 tb_news 中 的 图 书信 息 记录 按 图 书 序 号 进行 降序 排列 , 单 击 “ 降 序 排列 ”按钮 ， 
即 可 按 图 书 序号 进行 降序 排列 。〔 实 例 位 置 ， 光盘 \TM\Instance\20\20.20) 


图 20.26 ”查询 前 三 条 图 书信 息 记 录 


有 具体 步骤 如 下 。 

(1) 创建 数据 库 连 接 文件 conn.php。 其 代码 如 下 : 
<?php 
$conn=mysql_connect("localhost","root","111"); 
mysql_select_db("db_database20",$conn); 
mysql_query("set names utf8"); 

?> 


/| 连接 数 据 库 服务 器 
/| 连接 db_database20 数据 库 
// 设 置 数据 库 编码 格式 


(2) 创建 ndex.php 文件 ， 设 计 图 书信 息 查询 表单 页 面 ， 效 果 如 图 20.27 所 示 。 


(3) 完成 降序 排列 操作 。 其 代码 如 下 : 
<?php 
include("conn.php"); 
iflisset($_POST[Submit2]))f 
$select=mysql_query("select * from tb_book1",$conn); 
Srows=mysql_fetch_array($select); 
Jelse{ 


/包含 conn.php 文件 
// 判 断 按 钮 事件 为 空 时 
// 显 示 数 据 表 中 所 有 记录 


// 否 则 按 id 进行 降序 排列 ， 并 获取 返回 值 


$select=mysql_query("select * from tb_book1 order by id desc",$conn); 


$rows=mysql_fetch_array($select): 


} 

?> 

<?php 

dof 
?> 
<tr> 
<td><?php echo $rows[id"];?>&nbsp;</td> 
<td><?php echo $rows[bookname'];?>&nbsp;</td> 
<td><?php echo $rows['maker'];?>&nbsp;</td> 
<td><?php echo $rows['publisher];?>&nbsp;</td> 
<td><?php echo $rows[issuDate'];?>&nbsp;</td> 
<td><?php echo $rows[price'];?>&nbsp;</td> 
</tr> 
<?php 


/使 用 do while 循环 


<!- 输 出 id--> 
<!- 输 出 书 名 --> 
<!- 输 出 作者 --> 

< 上 -输出 出 版 社 -> 
<L- 输 出 出 版 时 间 --> 
< 上 -输出 价格 -> 
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}while($rows=mysql_fetch_array($select)); // 执 行 降序 排列 操作 


2 


运行 结果 如 图 20.27 所 示 。 


全 醒 #i 作 记性 理 大 过 
将 图 书信 息 按 ]D 进 行 降 序 排 列 


驳 
PNR 
i 


Pre 
A 
A 


图 20.27 将 图 书信 息 按 ID 降序 排列 
20.7 小 结 


本 章 首 先 介 绍 了 PHP 访问 MySQL 数据 库 的 一 般 流程 ， 然 后 详细 介绍 了 该 流程 每 一 步骤 的 具体 实现 方 
法 ， 并 重点 讲解 了 常用 函数 的 使 用 方法 ， 最 后 通过 公告 栏 实例 更 深层 次 地 介绍 了 PHP 如 何 实现 对 MySQL 
数据 库 进行 增 、 删 、 改 、 查 操作 。 通 过 本 章 的 学 习 ， 读 者 能 够 掌握 PHP 操作 MySQL 数据 库 的 一 般 流程 ， 
掌握 常用 MySQL 函数 的 使 用 方法 ， 并 能 够 具备 独立 完成 基本 数据 库 程序 的 能 力 。 和 希望 本 章 能 够 起 到 抛 砖 引 
玉 的 作用 ， 帮 助 读者 在 此 基础 上 更 深层 次 地 学 习 PHP 操作 MySQL 数据 库 的 相关 技术 ， 并 进一步 学 习 使 用 
面向 对 象 的 方式 操作 MySQL 数据 库 的 方法 。 


20.8 学 习 成 果 检 验 


1. 查询 结果 分 页 显示 ， 运 行 效果 如 图 20.28 所 示 。 (实例 位 置 ， 光盘 \TM\Instance\20\20.21) 
2. 按照 员工 编号 查询 员工 的 详细 信息 ， 运 行 效果 如 图 20.29 所 示 。《〔 实 例 位 置 ， 光盘 \TM\Instance\ 


20\20.22) 
查询 企业 员工 的 详细 信息 
一 一 一 
| 寺 | 如 名 | 电 En 
| i007 | | 1 埠 
图 20.28 查询 结果 分 页 显示 图 20.29 按 编号 查询 员工 信息 


3. 动态 显示 新 闻 信 息 ,， 截取 部 分 新 闻 主题 字符 串 , 屏蔽 乱码 。( 实 例 位置 : 光盘 \TMNInstance\20\20.23) 


PIE 
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( 种 视频 讲解 : 60 分 钟 ) 


在 PHP 的 早期 版 本 中 , 各 种 不 同 的 数据 库 扩展 (MySQL.M5S SQL、 
Oracle) 根本 没有 真正 的 一 致 性 ， 虽 然 都 可 以 实现 相同 的 功能 ， 但 是 
这 些 扩展 却 互 不 兼容 ,都 有 各 自 的 操作 函数 ,各自为政 。 结 果 导 致 PHP 
的 维护 非常 困难 ， 可 移植 性 也 非常 差 ， 为 了 解决 这 些 问题 ，PHP 的 开 
发 人 员 编写 了 一 种 轻型 、 便 利 的 APl 来 统一 各 种 数据 库 的 共性 ， 从 而 
达到 PHP 脚本 最 大 程度 的 抽象 性 和 兼容 性 ， 这 就 是 数据 库 抽象 层 。 而 
在 本 章 中 将 要 介绍 的 是 目前 PHP 抽象 层 中 最 为 流行 的 一 种 一 一 PDO 
抽象 层 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

MH 了 解 什么 是 PDO 

Wm 掌握 PDO 连接 数据 库 

MW 掌握 在 PDO 中 要 行 SQL 语句 

让 掌握 在 PDO 中 获取 结果 集 

H 掌握 在 PDO 中 捕获 SQL 语句 中 的 错误 

由 掌握 PDO 中 的 错误 处 理 

MW 掌握 PDO 中 的 事务 处 理 

MW 掌握 PDO 中 的 存储 过 程 
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21.1 什么 是 PDO 


21.1.1 PDO 概述 


PDO 是 PHP Date Object (PHP 数据 对 象 ) 的 简称 ， 它 是 与 PHP 5.1 版 本 一 起 发 行 的 ， 目 前 支持 的 数据 
库 包括 Firebird、FreeTDS、Interbase、MySQL、MS SQL Server、ODBC、Oracle、Postgre SQL、SQLite 和 
Sybase。 有 了 PDO， 不 必 再 使 用 mysql * 函 数 、oci_* 函 数 或 者 mssql * 函 数 ， 也 不 必 再 为 它们 封装 数据 库 操 
作 类 ， 只 需要 使 用 PDO 接口 中 的 方法 就 可 以 对 数据 库 进 行 操作 。 在 选择 不 同 的 数据 库 时 ， 只 需 修改 PDO 
的 DSN (数据 源 名 称 ) 。 

在 PHP 6 中 将 默认 使 用 PDO 连接 数据 库 ， 所 有 非 PDO 扩展 将 会 在 PHP 6 中 被 移 除 。 该 扩展 提供 PHP 
内 置 类 PDO 来 对 数据 库 进行 访问 ， 不 同 数据 库 使 用 相同 的 方法 名 ， 来 解决 数据 库 连接 不 统一 的 问题 。 


21.1.2 ”PDO 特点 


PDO 是 一 个 “数据 库 访问 抽象 层 ”， 作 用 是 统一 各 种 数据 库 的 访问 接口 ， 与 mysql 和 mssql 函数 库 相 
比 ，PDO 让 跨 数 据 库 的 使 用 更 具有 亲和力 ， 与 ADODB 和 MDB2 相 比 ，PDO 更 高 效 。 

PDO 将 通过 一 种 轻型 、 清 晰 、 方 便 的 函数 ， 统 一 各 种 不 同 RDBMS 库 的 共有 特性 ， 实 现 PHP 脚本 最 大 
程度 的 抽象 性 和 兼容 性 。 

PDO 吸取 现 有 数据 库 扩 展 成 功 和 失败 的 经 验 教 训 , 利用 PHP 5 的 最 新 特性 , 可 以 轻松 地 与 各 种 数据 库 
进行 交互 。 

PDO 扩展 是 模块 化 的 ， 能 够 在 运行 时 为 你 的 数据 库 后 端 加 载 驱动 程序 ， 而 不 必 重 新 编译 或 重新 安装 整 
个 PHP 程序 。 例 如 ，PDO_MySQL 扩展 会 蔡 代 PDO 扩展 实现 MySQL 数据 库 API。 还 有 一 些 用 于 
Oracle、PostgreSQL、ODBC 和 Firebird 的 驱动 程序 ， 更 多 的 驱动 程序 尚 在 开发 中 。 


21.1.3 安装 PDO 


PDO 是 与 PHP 5.1 一 起 发 行 的 ， 默 认 包含 在 PHP 5.1 中 。 由 于 PDO 需要 PHP 5 核心 面向 对 象 特性 的 支 
持 ， 因 此 其 无 法 在 PHP 5.0 之 前 的 版 本 中 使 用 。 

默认 情况 下 ，PDO 在 PHP 5.2 中 为 开启 状态 ， 但 是 要 启用 对 某 个 数据 库 驱 动 程序 的 支持 ， 仍 需要 进行 
相应 的 配置 操作 。 

在 Linux 环境 下 ， 要 使 用 MySQL 数据 库 ， 可 以 在 configure 命令 中 添加 如 下 选项 : 

~--with-pdo-mysql=/path/to/mysqlinstallation 

在 Windows 环境 下 ，PDO 在 php.ini 文件 中 进行 配置 ， 如 图 21.1 
所 示 


要 启用 PDO， 首 先 必须 加 载 “extension=php_pdo.dll”， 如 果 要 想 
其 支持 某 个 具体 的 数据 库 ， 那 么 还 要 加 载 对 应 的 数据 库 选 项 。 例 如 ， 

要 支持 MySQL 数据 库 ， 则 需要 加 载 “extension=php_pdo_mysql.dll” 
选项 。 


[CN 


21.1 Windows 环境 下 配置 PDO 
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和 起 在 完成 数据 库 的 加 载 后 ， 要 保存 php.ini 文件 ， 并 且 重 新 启动 Apache 服务 器 ， 修 改 即 可 生效 。 


21.2 PDO 连接 数据 库 


峰 1 视频 讲解 : 光盘 \TM\Video\ 第 21 章 \PDO 连接 数据 库 .exe 
21.2.1 PDO 构造 函数 


在 PDO 中 ， 要 建立 与 数据 库 的 连接 需要 实例 化 PDO 的 构造 函数 。PDO 构造 函数 的 语法 如 下 : 
__construct(string $dsn[,string $usernamel[,string $password[,array $driver_options]]]) 

参数 说 明 如 下 。 

dsn: 数据 源 名 ， 包 括 主机 名 端口 号 和 数据 库 名 称 。 

username: 连接 数据 库 的 用 户 名 。 

password: 连接 数据 库 的 密码 。 

driver_options: 连接 数据 库 的 其 他 选项 。 

通过 PDO 连接 MySQL 数据 库 的 代码 如 下 : 


<?php 
header("Content-Type:text/html;charset=utf-8"); // 设 置 页 面 的 编码 格式 
$dbms='mysql': /| 数据 库 类 型 
$dbName='db_database21'; // 使 用 的 数据 库 名 称 
$user='root'; // 使 用 的 数据 库 用 户 名 
S$pwd="111'; // 使 用 的 数据 库 密码 
$host='localhost'; // 使 用 的 主机 名 称 
$dsn="$dbms:host=$host;dbname=$dbName"; 
try{ /捕获 异常 
$pdo=new PDO($dsn,$user,$pwd); /实例 化 对 象 
echo "PDO 连接 MySQL 成 功 "; 
} catch (Exception $e) { 
echo $e->getMessage()."<br>"; 
} 
?> 


21.2.2 DSN 详解 


DSN 是 Data Source Name( 数 据 源 名 称 ) 的 首 字母 缩写 。 它 提供 连接 数据 库 需 要 的 信息 。PDO 的 DSN 
包括 3 部 分 : PDO 驱动 名 称 ( 如 mysql、sqlite 或 者 pgsql) 、 冒 号 和 驱动 特定 的 语法 。 每 种 数据 库 都 有 其 特 
定 的 驱动 语法 。 

在 使 用 不 同 的 数据 库 时 ， 必 须 明 确 数据 库 服务 器 是 完全 独立 于 PHP 的 实体 。 虽 然 笔者 在 讲解 本 书 的 内 
容 时 ， 数 据 库 服务 器 和 Web 服务 器 是 在 同一 台 计算 机 上 ， 但 是 实际 的 情况 可 能 不 是 如 此 。 数 据 库 服务 器 可 
能 与 Web 服务 器 不 是 在 同一 台 计算 机 上 ,此 时 要 通过 PDO 连接 数据 库 时 , 就 需要 修改 DSN 中 的 主机 名 称 。 

由 于 数据 库 服务 器 只 在 特定 的 端口 上 监听 连接 请 求 。 而 每 种 数据 库 服务 器 具有 一 个 默认 的 端口 号 

(MySQL 是 3306) ， 数 据 库 管 理 员 又 可 以 对 端口 号 进行 修改 ， 因 此 有 可 能 PHP 找 不 到 数据 库 的 端口 ， 此 
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时 就 可 以 在 DSN 中 包含 端口 号 。 
另外 由 于 一 个 数据 库 服务 器 中 可 能 拥有 多 个 数据 库 ， 所 以 在 通过 DSN 连接 数据 库 时 ， 通 常 都 包括 数据 
库 名 称 ， 这 样 可 以 确保 连接 的 是 你 想 要 的 数据 库 ， 而 不 是 其 他 的 数据 库 。 


21.3 PDO 中 执行 SQL 语句 


贸 中 视频 讲解 : 光盘 \TM\Video\ 第 21 章 \PDO 中 执行 SQL 语句 .exe 
在 PDO 中 ， 可 以 使 用 下 面 的 3 种 方法 来 执行 SQL 语句 。 


21.3.1 exec() 方 法 
exec0 方 法 返回 执行 后 受 影响 的 行 数 。 其 语法 如 下 : 
int PDO::exec ( string statement ) 
参数 statement 是 要 执行 的 SQL 语句 。 
该 方法 返回 执行 查询 时 受 影 响 的 行 数 ， 通 常用 于 INSERT、DELETE 和 UPDATE 语句 中 。 


2 


ik 


.3.2 query() 方 法 

query0 方 法 通过 用 于 返回 执行 查询 后 的 结果 集 。 其 语法 如 下 : 
PDOStatement PDO::query ( string statement ) 

参数 statement 是 要 执行 的 SQL 语句 。 它 返回 的 是 一 个 PDOStatement 对 象 。 
21.3.3” 预 处 理 语句 一 一 prepare() 和 execute() 


预 处 理 语句 包括 prepare0 和 execute0 两 个 方法 。 首 先 ， 通 过 prepare0 方 法 做 查询 的 准备 工作 ， 然 后 ， 通 
过 execute0 方 法 执行 查询 。 并 且 还 可 以 通过 bindParam() 方 法 来 绑 定 参数 提供 给 execute0 方 法 。 其 语法 如 下 : 

PDOStatement PDO::prepare ( string statement [, array driver_options] ) 

bool PDOStatement::execute ( [array input_parameters] ) 


21.4 PDO 中 获取 结果 集 


多 1 视频 讲解 : 光盘 \TM\Video\ 第 21 章 \PDO 中 获取 结果 集 .exe 
在 PDO 中 获取 结果 集 有 3 种 方法 : fetch())、fetchAll0 和 fetchColumn0。 


21.4.1 fetch() 方 法 


fetch0 方 法 获取 结果 集中 的 下 一 行 。 其 语法 格式 如 下 : 

mixed PDOStatement::fetch ( [int fetch_style [, int cursor_orientation [, int cursor_offset]] ) 
参数 说 明 如 下 : 

fetch_style: 控制 结果 集 的 返回 方式 ， 其 可 选 方式 如 表 21.1 所 示 。 
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表 21.1 fetch_style 控制 结果 集 的 可 选 值 


值 说 明 
PDO::FETCH ASSOC | 关联 数组 形式 
PDO:FETCH NUM | ”数字 索引 数组 形式 
PDO:FETCH BOTH | ” 两 者 的 数组 形式 都 有 有， 这 是 默认 的 
PDO::FETCH OBJ 按照 对 象 的 形式 ， 类 似 于 以 前 的 mysql_fetch_objectO 


PDO:EETCH BOUND 以 布尔 值 的 形式 返回 结果 ， 同 时 将 获取 的 列 值 赋 给 bindParam( 方 法 中 指定 的 变量 


PDO::FETCH LAZY 以 关联 数组 、 数 字 索 引 数组 和 对 象 3 种 形式 返回 结果 


cursor_orientation: PDOStatement 对 象 的 一 个 滚动 游标 ， 可 用 于 获取 指定 的 一 行 。 
回 “cursor_ offset:， 游标 的 偏 移 量 。 
例 21.1 通过 fetch0 方 法 获取 结果 集中 下 一 行 的 数据 , 进而 应 用 while 语句 完成 数据 库 中 数据 的 循环 输 
出 。〔 实 例 位置 ， 光盘 \TM\Instance\2121.1) 
创建 index.php 文件 ， 设 计 网 页 页 面 。 首先 ， 通 过 PDO 连接 MySQL 数据 库 。 然 后 ， 定 义 SELECT 查询 
语句 ， 应 用 prepare() 和 execute0 方 法 执行 查询 操作 。 接 着 ,通过 fetch0 方 法 返回 结果 集中 下 一 行 数据 ， 同 时 
设置 结果 集 以 关联 数组 形式 返回 。 最 后 ， 通 过 while 语句 完成 数据 的 循环 输出 。 其 关键 代码 如 下 : 
<?php 
$dbms='mysql': /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那 么 多 的 函数 
$host="localhost'; // 数 据 库 主机 名 
$dbName='db_database21'; // 使 用 的 数据 库 
$user='root’; /数据 库 连 接 用 户 名 
S$pass="111"; // 对 应 的 密码 
$dsn="$dbms:host=$host;dbname=$dbName"; 
try{ 
$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连 接 对 象 $pdo 
$query="select * from tb_pdo_mysql"; /定义 SQL 语句 
Sresult=$pdo->prepare($query); /准备 查询 语句 
$result->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 
while($res=$result->fetch(PDO::FETCH_ASSOC)){ /循环 输出 查询 结果 集 ， 并 且 设 置 结果 集 的 为 关联 索引 
?> 
<tr> 
<td height="22" align="center" valign="middle"><?php echo $res[id];?></td> 
<td align="center" valign="middle"><?php echo S$res['pdo_type'];?></td> 
<td align="center" valign="middle"><?php echo $res['database_name'];?></td> 
<td align="center' valign="middle"><?php echo $res['dates'];?></td> 
<td align="center" valign="middle"> <a href="#"> 删 除 </a></td> 


<?php 


1 
} catch (PDOException $e){ 
die ("Error!: " . $e->getMessage() . "<br/>"); 


Te 
运行 结果 如 图 21.2 所 示 。 


21.4.2 ”fetchAll() 方 法 


fetchAl10 方 法 获取 结果 集中 的 所 有 行 。 其 语法 如 下 : 
array PDOStatement::fetchAll ( [int fetch_style [, int column_index]] ) 
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参数 说 明 如 下 : 

fetch_style: 控制 结果 集中 数据 的 显示 方式 。 
column index: 字段 的 索引 。 

其 返回 值 是 一 个 包含 结果 集中 所 有 数据 的 二 维 数组 。 


例 21.2 通过 fetchAl10 方 法 获取 结果 集中 所 有 行 ， 并 且 通 过 for 语句 读 取 二 维 数组 中 的 数据 ， 完 成 数 


据 库 中 数据 的 循环 输出 。《〈 实 例 位 置 ; 光盘 \TMNInstance\2l\21.2) 
创建 mdex.php 文件 ， 设 计 网 页 页 面 。 首 先 ， 通 过 PDO 连接 MySQL 数据 库 。 然 后 ， 


定义 SELECT 查询 


语句 ， 应 用 prepare() 和 execute0 方 法 执行 查询 操作 。 接 着 ， 通 过 fetchAll0 方 法 返回 结果 集中 所 有 行 。 最 后 ， 


通过 for 语句 完成 结果 集中 所 有 数据 的 循环 输出 。 其 关键 代码 如 下 : 


<?php 
$dbms='mysql'; /| 数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 
$host="localhost’; /数据 库 主机 名 
$dbName='db_database21'; // 使 用 的 数据 库 
$user='root’; /数据 库 连 接 用 户 名 
S$pass="111'; // 对 应 的 密码 
S$dsn="$dbms:host=$host;dbname=$dbName"; 
try{ 
$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连接 对 象 $pdo 
$query="select * from tb_pdo_mysql"; /定义 SQL 语句 
$result=$pdo->prepare($query); // 准 备查 询 语句 
$result->execute(); /执行 查询 语句 ， 并 返回 结果 集 
$res=$result->fetchAll(PDO::FETCH_ASSOC); 1/ 获取 结果 集中 的 所 有 数据 
for($i=0;$i<count($res);$i++X{ // 循 环 读 取 二 维 数组 中 的 数据 
?> 
<tr> 


<td height="22" align="center" valign="middle"><?php echo $res[$il[id];?></td> 


<td align="center" valign="middle"><?php echo $res[$il[pdo_type]?></td> 


<td align="center valign="middle"><?php echo $res[$il[database_name'];?></td> 


<td align="center" valign="middle"><?php echo $res[$il['dates'];?></td> 
<td align="center" valign="middle"> <a href="#"> 删 除 </a></td> 
</tr> 
<?php 


; 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage() . "<br/>"); 
} ?> 
运行 结果 如 图 21.3 所 示 。 
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21.4.3 ”fetchColumn() 方 法 


fetchColumn0 方 法 获取 结果 集中 下 一 行 指定 列 的 值 。 其 语法 如 下 : 

string PDOStatement::fetchColumn ( [int column_number] ) 

可 选 参数 column_number 设置 行 中 列 的 索引 值 ， 该 值 从 0 开始 。 如 果 省 略 该 参数 则 将 从 第 1 列 开始 取 值 。 

通过 fetchColumn0 方 法 获取 结果 集中 下 一 行 中 指定 列 的 值 , 注意 , 这 里 是 “结果 集中 下 一 行 中 指定 列 的 值 ”。 

例 21.3 ”本 实例 输出 数据 表 中 第 一 列 的 值 ， 即 输出 数据 的 ID。( 实 例 位 置 : 光盘 \TM\Instance2121.3) 

创建 index.php 文件 ， 设 计 网 页 页 面 。 首先 ， 通 过 PDO 连接 MySQL 数据 库 。 然 后 ， 定 义 SELECT 查询 
语句 ， 应 用 prepare0 和 execute0 方 法 执行 查询 操作 。 接 着 ， 通 过 fetchColumn() 方 法 输出 结果 集中 下 一 行 第 
一 列 的 值 。 其 关键 代码 如 下 。 


<?php 
$dbms="'mysql'; /| 数据库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 
$host='localhost'; /数据 库 主机 名 
$dbName='db_database21'; // 使 用 的 数据 库 
$user='root’; /| 数据库 连接 用 户 名 
$pass='111'; // 对 应 的 密码 
$dsn="$dbms:host=$host;dbname=$dbName"; 
try{ 
$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连 接 对 象 $pdo 
$query="select * from tb_pdo_mysql"; /定义 SQL 语句 
$result=$pdo->prepare($query); // 准 备查 询 语句 
$result->execute(); // 执 行 查询 语句 ， 并 返回 结果 集 
> 
<tr> 
<td height="22" align="center" valign="middle"><?php echo $result->fetchColumn(0);?></td> 
</tr> 
<tr> 
<td height="22" align="center" valign="middle"><?php echo $result->fetchColumn(0);?></td> 
</tr> 
<tr> 
<td height="22" align="center" valign="middle"><?php echo $result->fetchColumn(0);?></td> 
</tr> 
<tr> 
<td height="22" align="center" valign="middle"><?php echo $result->fetchColumn(0);?></td> 
</tr> 
<?php 


} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage() . "<br/>"); 


?> 
运行 结果 如 图 21.4 所 示 。 


了 DC 第 一 列 , 数 峰 ID 值 》 


21.4 ”fetchColumn0 方 法 获取 结果 集中 第 一 列 的 值 
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21.5 PDO 中 捕获 SQL 语句 中 的 错误 


视频 讲解 : 光盘 \TM\Video\ 第 21 章 \PDO 中 捕获 SQL 语句 中 的 错误 .exe 
在 PDO 中 捕获 SQL 语句 中 的 错误 有 3 种 方案 可 以 选择 。 


21.5.1 使 用 默认 模式 


PDO::ERRMODE_SILENT 


在 默认 模式 中 设置 PDOStatement 对 象 的 errorCode 属性 ， 但 不 进行 其 他 任何 操作 。 

例 21.4 通过 prepare0 和 execute0 方 法 向 数据 库 中 添加 数据 ,设置 PDOStatement 对 象 的 errorCode 属性 ， 
手动 检测 代码 中 的 错误 。《〈 实 例 位 置 : 光盘 \TM\Instance21\21.4) 

创建 index.php 文件 ， 添 加 form 表单 ， 将 表单 元 素 提交 到 本 页 。 通 过 PDO 连接 MySQL 数据 库 ， 通 过 
预 处 理 语句 prepare() 和 execute0 执 行 INSERT 添加 语句 ， 向 数据 表 中 添加 数据 ， 并 且 设 置 PDOStatement 对 
象 的 errorCode 属性 ， 检 测 代码 中 的 错误 。 其 关键 代码 如 下 。 

<?phl 

ema &8 $_POST[pdoll="){ 

$dbms='mysql'; /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 


$host='localhost'; /数据 库 主 机 名 

$dbName='db_database21'; /使 用 的 数据 库 

$user='root'; /数据 库 连 接 用 户 名 

$pass='111"; // 对 应 的 密码 
$dsn="$dbms:host=$hostidbname=$dbName'"'; 
$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连 接 对 象 $pdo 


$query="insert into 
tb_pdo_mysqls(pdo_type,database_name,dates)values(".$_POST[pdo].",".$_POST[databases'].",".$_POST 
[dates].")”; 
Sresult=$pdo->prepare($query); 
Sresult->execute(); 
$code=$result->errorCode(); 
iempty($code))f 
echo "数据 添加 成 功 !"; 
jelsef 
echo 数据 库 错 误 : <br/>'; 
echo 'SQL Query:'.$query; 
echo “'<pre>' 
var_dump($result->errorlnfo()); 
echo '</pre>"; 
} 
上 


?> 


在 本 实例 中 ， 在 定义 INSERT 添加 语句 时 ， 使 用 了 错误 的 数据 表 名 称 tb_pdo_mysqls (正确 名 称 是 
tb_pdo_ mysql) ， 导 致 输出 结果 如 图 21.5 所 示 。 


21.5.2 ”使 用 警告 模式 


PDO::ERRMODE_WARNING 


警告 模式 会 产生 一 个 PHP 警告 ,并 设置 errorCode 属性 。 如 果 设置 的 是 警告 模式 ， 那 么 除非 明确 地 检查 
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错误 代码 ， 否 则 程序 将 继续 按照 其 方式 运行 。 
例 21.5 设置 警告 模式 ， 通 过 prepare0 和 execute0 方 法 读 取 数据 库 中 数据 ， 并 且 通 过 while 语句 和 fetch0 方 
法 完成 数据 的 循环 输出 , 体会 在 设置 成 警告 模式 后 执行 错误 的 SQL 语句 。 (实例 位 置 : 光盘 \TM\Instance21\21.5) 
创建 mdex.php 文件 ， 连 接 MySQL 数据 库 ， 通 过 预 处 理 语句 prepare0 和 execute0 执 行 SELECT 查询 语 
句 ,并 设置 一 个 错误 的 数据 表 名 称 , 同时 通过 setAttribute0 方 法 设置 为 警告 模式 , 最 后 通过 while 语句 和 fetchO 
方法 完成 数据 的 循环 输出 。 其 关键 代码 如 下 : 


<?php 
$dbms=imysql'; ”// 数 据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 
$host="localhost'; /数据 库 主 机 名 
$dbName='db_database21'; // 使 用 的 数据 库 
$user= root'; /数据 库 连 接 用 户 名 
$pass="111'; // 对 应 的 密码 
$dsn="$dbms:host=$host;dbname=$dbName"; 
try{ 
$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连接 对 象 $pdo 
S$pdo->setAttribute(PDO::ATTR_ERRMODE,PDO::ERRMODE_WARNING); // 设 置 为 警告 模式 
$query="select * from tb_pdo_mysqls"; /定义 SQL 语句 
$result=$pdo->prepare($query); /准备 查询 语句 
$result->execute(); /执行 查询 语句 ， 并 返回 结果 集 
while($res=$result->fetch(PDO::FETCH_ASSOC)X /while 循环 输出 查询 结果 集 ， 并 且 设置 结果 集 的 为 关联 索引 
Ee 
<tr> 
<td height="22" align="center" valign="middle"><?php echo S$res[id'"];?></td> 
<td align="center" valign="middle"><?php echo S$res['pdo_type'];?></td> 
<td align="center" valign="middle"><?php echo S$res['database_name'];?></td> 
<td align="center" valign="middle"><?php echo S$res['dates'"];?></td> 
</tr> 
<?php 
} 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage() . "<br/>"); 
上 
?> 


在 设置 为 警告 模式 后 ， 如 果 SQL 语句 出 现 错误 将 给 出 一 个 提示 信息 ， 但 是 程序 仍 能 够 继续 执行 下 去 ， 
其 运行 结果 如 图 21.6 所 示 。 


J 


图 21.5 在 默认 模式 中 捕获 SQL 中 的 错误 图 21.6 设置 警告 模式 后 捕获 的 SQL 语句 错误 
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21.5.3 ”使 用 异常 模式 PDO::ERRMODE_EXCEPTION 


异常 模式 会 创建 一 个 PDOException， 并 设置 errorCode 属性 。 它 可 以 将 执行 代码 封装 到 一 个 try…catch 
语句 块 中 。 未 捕获 的 异常 将 会 导致 脚本 中 断 ， 并 显示 堆栈 跟踪 让 你 了 解 是 哪里 出 现 的 问题 。 

例 21.6 在 执行 数据 库 中 数据 的 删除 操作 时 , 设置 为 异常 模式 ， 
并 且 编 写 一 个 错误 的 SQL 语句 (操作 错误 的 数据 表 tb_pdo_mysqls)， 
体会 异常 模式 与 警告 模式 和 默认 模式 的 区 别 。( 实 例 位 置 : 光盘 \TM 
Instance\21\21.6) 


(1) 创建 index.php 文件 ,连接 MySQL 数据 库 , 通过 预 处 理 语 Ve | 
句 prepare0 和 execute() 执 行 SELECT 查询 语句 , 再 通过 while 语句 和 4 ee 


fetch0 方 法 完成 数据 的 循环 输出 ， 并 且 设 置 删除 超 链 接 ， 链 接 到 
delete.php 文件 ， 传 递 的 参数 是 数据 的 ID 值 。 其 运行 效果 如 图 21.7 
所 示 。 

(2) 创建 delete php 文件 ， 获 取 超 链接 传递 的 数据 ID 值 。 连接 人 
数据 库 ， 通 过 setAttribute(0) 方 法 设置 为 异常 模式 ， 定 义 DELETE 删除 语句 ， 删 除 一 个 错误 数据 表 

(tb _pdo_ mysqls) 中 的 数据 ， 并 且 通 过 try…catch 语句 捕获 错误 信息 。 其 代码 如 下 : 

<?php 

header ( "Content-type: text/html; charset=utf-8" ); /设置 文件 编码 格式 

if$_GET[conn_id]i=jf 

$dbms='mysql'; /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 


$host="localhost'; /数据 库 主 机 名 
$dbName='db_database21'; /使 用 的 数据 库 
$user='root'; /数据 库 连 接 用户 名 
S$pass="111"; // 对 应 的 密码 


$dsn="$dbms:host=$hostidbname=$dbName'"' 


$pdo = new PDO($dsn, $user, $pass); /初始化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连接 对 象 $pdo 
$pdo->setAttribute(PDO::ATTR_ERRMODE,PDO::ERRMODE_EXCEPTION); 
$query="delete from tb_pdo_mysqls where Id=:id"; 
$result=$pdo->prepare($query); // 预 准备 语句 
$result->bindParam(":id',$_GET[conn_id]);  // 绑 定 更 新 的 数据 
Sresult->execute(); 

} catch (PDOException $e){ 
echo 'PDO Exception Caught."; 
echo 'Error with the database:<br/>"; 
echo 'SQL Query: '.$query; 
echo '<pre>'; 

echo "Error: " . $e->getMessage(). "<br/>"; 
echo "Code: " . $e->getCode(). "<br/>"; 
echo "File: " . $e->getFile(). "<br/>"; 
echo "Line: " . $e->getLine(). "<br/>"; 
echo "Trace: " . $e->getTraceAsString(). "<br/>"; 
echo '</pre>"; 


} 


?> 


在 设置 为 异常 模式 后 ， 执 行 错误 的 SQL 语句 返回 的 结果 如 图 21.8 所 示 。 
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图 21.8 异常 模式 捕获 的 SQL 语句 错误 信息 


21.6 PDO 中 的 错误 处 理 


区 Nd 视频 讲解 : 光盘 \TM\Video\ 第 21 章 \PDO 中 错误 处 理 .exe 
在 PDO 中 有 两 个 获取 程序 中 错误 信息 的 方法 : errorCode0 方 法 和 errorInfo0 方 法 。 


21.6.1 errorCode() 方 法 


码 。 


errorCode() 方 法 用 于 获取 在 操作 数据 库 句 柄 时 所 发 生 的 错误 代码 ， 这 些 错误 代码 被 称 为 SQLSTATE 代 
其 语法 格式 如 下 : 

int PDOStatement::errorCode ( void ) 

errorCode() 方 法 返回 一 个 SQLSTATE，SQLSTATE 是 由 5 个 数字 和 字母 组 成 的 代码 。 

例 21.7 在 PDO 中 通过 query() 方 法 完成 数据 的 查询 操作 , 并 且 通 过 foreach 语句 完成 数据 的 循环 输出 。 


在 定义 SQL 语句 时 使 用 一 个 错误 的 数据 表 , 并 且 通 过 errorCode0 方 法 返回 错误 代码 。( 实 例 位 置 : 光盘 \TM 
Instance\21\21.7) 


创建 index.php 文件 。 首 先 通过 PDO 连接 MySQL 数据 ,然后 通过 query0 方 法 执行 查询 语句 ,接着 通过 


errorCode() 方 法 获取 错误 代码 ， 最 后 通过 foreach 语句 完成 数据 的 循环 输出 。 其 关键 代码 如 下 。 


<?php 
$dbms='mysql; ”// 数 据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 
$host='localhost'; /数据 库 主机 名 
$dbName='db_database21'; /使 用 的 数据 库 
$user=root'; /数据 库 连 接 用 户 名 
$pass="111'; // 对 应 的 密码 
$dsn="$dbms:host=$host;dbname=$dbName"; 
try{ 
$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连接 对 象 $pdo 
$query="select * from tb_pdo_mysqls"; /定义 SQL 语句 
Sresult=$pdo->query($query); /| 执行 查询 语句 ， 并 返回 结果 集 
echo "errorCode 为 : "$pdo->errorCode(); 
foreach($result as $items}{ 
i 


<tr> 
<td height="22" align="center" valign="middle"><?php echo $items[id];?></td> 
<td align="center" valign="middle"><?php echo S$items[pdo_type'];?></td> 
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<td align="center valign="middle"><?php echo S$items['database_name'];?></td> 
<td align="center valign="middle"><?php echo $items['dates];?></td> 

</tr> 
<?php 


I 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage() . "<br/>"); 
4 


人 
运行 结果 如 图 21.9 所 示 。 


21.6.2 errorlnfo() 方 法 


errorInfo() 方 法 用 于 获取 操作 数据 库 句 柄 时 所 发 生 的 错误 信息 。 其 语法 格式 如 下 : 

array PDOStatement::errorInfo ( void ) 

errorInfo0) 方 法 的 返回 值 为 一 个 数组 ， 它 包含 了 相关 的 错误 信息 。 

例 21.8 在 PDO 中 通过 query0 方 法 完成 数据 的 查询 操作 , 并 且 通 过 foreach 语句 完成 数据 的 循环 输出 。 
在 定义 SQL 语句 时 使 用 一 个 错误 的 数据 表 ， 并 且 通 过 errorInfo0 方 法 返回 错误 信息 。( 实 例 位 置 : 光盘 \TM\ 
Instance\21\21.8) 

创建 index.php 文件 。 首先 通过 PDO 连接 MySQL 数据 库 , 然后 通过 query() 方 法 执行 查询 语句 ， 接 着 通 
过 errorInfo0 方 法 获取 错误 信息 ， 最 后 通过 foreach 语句 完成 数据 的 循环 输出 。 其 关键 代码 如 下 : 


<?php 
$dbms='mysql; /数据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 
$host="localhost'; /数据 库 主机 名 
$dbName='db_database21'; // 使 用 的 数据 库 
$user='root'; /数据 库 连接 用 户 名 
S$pass="111'; // 对 应 的 密码 
$dsn="$dbms:host=$host;dbname=$dbName"; 
try{ 
$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连接 对 象 $pdo 
$query="select * from tb_pdo_mysqls"; /定义 SQL 语句 
Sresult=$pdo->query($query); /执行 查询 语句 ， 并 返回 结果 集 
print_r($pdo->errorlnfo()); 
foreach($result as $itemsj{ 
?> 
<tr> 


<td height="22" align="center" valign="middle"><?php echo $items[id];?></td> 
<td align="center" valign="middle"><?php echo S$items[pdo_type'];?></td> 

<td align="center" valign="middle"><?php echo S$items['database_name'];?></td> 
<td align="center" valign="middle"><?php echo S$items['dates'];?></td> 


</tr> 
<?php 
} 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage() . "<br/>"); 
} 
?> 


运行 结果 如 图 21.10 所 示 。 
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图 21.9 通过 errorCode(0 方 法 获取 错误 代码 图 21.10 通过 errorInfo0 方 法 获取 错误 信息 


21.7 PDO 中 的 事务 处 理 


名 0 视频 讲解 :光盘 \TM\Video\ 第 21 章 \PDO 中 事务 处 理 .exe 

在 PDO 中 同样 可 以 实现 事务 处 理 的 功能 ， 其 应 用 的 方法 如 下 : 

开启 事务 一 一 beginTransaction() 方 法 

beginTransaction() 方 法 将 关闭 自动 提交 (autocommit) 模式 ， 直 到 事务 提交 或 者 回 滚 以 后 才 恢复 。 

提交 事务 一 一 commit0 方 法 

commit0 方 法 完成 事务 的 提交 操作 ， 成 功 则 返回 TRUE， 和 否则 返回 FALSE。 

事务 回 深 一 一 rollback0 方 法 

rollback0) 方 法 执行 事务 的 回 滨 操 作 。 

例 21.9 通过 prepare0 和 execute0 方 法 向 数据 库 中 添加 数据 ， 并 且 通 过 事务 处 理 机 制 确保 数据 能 够 正 
确 地 添加 到 数据 中 。 实例 位 置 ， 光盘 \TMNVInstance\21\21.9) 

创建 index.php 文件 。 首 先 ， 定 义 数据 库 连接 的 参数 ， 创 建 ty…catch 语句 ， 在 try 语句 中 实例 化 PDO 
构造 函数 , 完成 与 数据 库 的 连接 , 并 且 通 过 beginTransaction0 方 法 开启 事务 。 然后, 定义 INSERT 添加 语句 ， 
通过 $_POST[] 方 法 获取 表单 中 提交 的 数据 ， 通 过 prepare0 和 execute() 方 法 向 数据 库 中 添加 数据 ， 并 且 通 过 
commit() 方 法 完成 事务 的 提交 操作 。 最 后 ， 在 catch 语句 中 返回 错误 信息 ， 并 且 通 过 rollBackO 执 行事 务 的 回 
滚 操 作 。 其 代码 如 下 : 

<?php 

if($_POST[Submit]==" 提 交 " && $_POST[pdoli="){ 

$dbms='mysql'; // 数 据 库 类 型 ， 对 于 开发 者 来 说 ,使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 


$host="localhost'; /数据库 主机 名 

SdbName="'db_database21'; // 使 用 的 数据 库 

$user='root'; /| 数据 库 连 接 用 户 名 

S$pass="111'; // 对 应 的 密码 

$dsn="$dbms:host=$host;dbname=$dbName"; 

try{ 

$pdo = new PDO($dsn, $user, $pass); // 初 始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连 接 对 象 $pdo 
$pdo->beginTransaction(); // 开 启事 务 


$query="inser into tb_pdo_mysql(pdo_type,database_name,dates)values(".$_POST[pdo].",".$_POST 
[databases]",".$_POST[dates].")"; 
Sresult=$pdo->prepare($query); 
if($result->execute()){ 
echo "数据 添加 成 功 !"; 
}else{ 


-加 
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echo "数据 添加 失败 !"; 


1 

$pdo->commit(); /执行 事务 的 提交 操作 
} catch (PDOException $e){ 

die ("Error!: " . $e->getMessage() . "<br/>"); 
S$pdo->rollBack(); // 执 行事 务 的 回 滚 
} 


GA 


行 结果 如 图 21.11 所 示 。 
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21.8 PDO 中 的 存储 过 程 


名 gl 视频 讲解 :光盘 \TM\Video\ 第 21 章 \PDO 中 存储 过 程 .exe 

存储 过 程 允许 在 更 接近 于 数据 的 位 置 操作 数据 ， 从 而 减少 带宽 的 使 用 ， 它 们 使 数据 独立 于 脚本 逻辑 ， 
允许 使 用 不 同 语言 的 多 个 系统 以 相同 的 方式 访问 数据 ， 从 而 节省 了 花费 在 编码 和 调试 上 的 宝贵 时 间 。 同 时 
它 使 用 预定 义 的 方案 执行 操作 ， 提 高 查询 速度 ， 并 且 能 够 阻止 与 数据 的 直接 相互 作用 ， 从 而 起 到 保护 数据 
的 作用 。 

下 面 讲解 如 何在 PDO 中 调用 存储 过 程 。 这 里 首先 创建 一 个 存储 过 程 ， 其 SQL 语句 如 下 : 

drop procedure if exists pro_reg; 

delimiter // 


create procedure pro_reg (in nc varchar(80), in pwd varchar(80), in email varchar(80),in address varchar(50)) 
begin 

insert into tb_reg (name, pwd ,email ,address) values (nc, pwd, email, address); 

end; 

I 


回 drop 语句 删除 MySQL 服务 器 中 已 经 存在 的 存储 过 程 pro_reg。 

delimiter /的 作用 是 将 语句 结束 符 更 改 为 “/”。 

in nc varchar(50)…in address varchar(50) 表 示 要 向 存储 过 程 中 传 入 的 参数 。 

begin…end 表示 存储 过 程 中 的 语句 块 ， 它 的 作用 类 似 与 PHP 语言 中 的 “{…}” 

存储 过 程 创建 成 功 后 ， 下 面 调用 这 个 存储 过 程 实现 用 户 注册 的 功能 。 

例 21.10 在 PDO 中 通过 CALL 语句 调用 存储 过 程 ， 实 现 用 户 注册 信息 的 添加 操作 。〔 实 例 位 置 ， 光 
盘 \TMNInstance\21\21.10) 

创建 index.php 文件 。 首 先 ， 创 建 form 表单 ， 将 用 户 注册 信息 通过 POST 方法 提交 到 本 页 。 然 后 ， 在 本 


局 
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页 中 编写 PHP 脚本 ,通过 PDO 连接 MySQL 数据 库 ， 并 且 设 置 数据 库 编码 格式 为 utf-8， 获 取 表 单 中 提交 的 
用 户 注册 信息 。 接 着 ， 通 过 call 语句 调用 存储 过 程 pro_reg， 将 用 户 注册 信息 添加 到 数据 表 中 。 最 后 ， 通 过 
try…catch 语句 块 返回 错误 信息 。 其 关键 代码 如 下 : 
<?php 
if($_POST[submit]!=™"X{ 
$dbms= mysql; ”// 数 据 库 类 型 ， 对 于 开发 者 来 说 ， 使 用 不 同 的 数据 库 ， 只 要 改 这 个 ， 不 用 记 住 那么 多 的 函数 
名 


$host="localhost'; /数据 库 主机 : 

S$dbName='db_database21'; /使 用 的 数据 库 

$user='root'; /数据 库 连接 用 户 名 

S$pass="111'; // 对 应 的 密码 

S$dsn="$dbms:host=$host;dbname=$dbName"; 

try{ 

$pdo = new PDO($dsn, $user, $pass); /初始 化 一 个 PDO 对 象 ， 就 是 创建 了 数据 库 连 接 对 象 
$pdo 

$pdo->query("set names utf8"); // 设 置 数据 库 编码 格式 


$pdo->setAttribute(PDO::ATTR_ERRMODE,PDO::ERRMODE_EXCEPTION); /定义 错误 异常 模式 
$nc=$_POST[nc]; 
S$pwd=md5($_POST[pwd]); 
$email=$_POST[email]; 
S$address=$_POST['address']; 

$query="call pro_reg('$nc','$pwd','$email','$address')"; 

Sresult=$pdo->prepare($query); 

if($result->execute()X{ 

echo "数据 添加 成 功 !"; 


Jelse{ 
echo "数据 添加 失败 !"; 


> 

} catch (PDOException $e) { 

echo 'PDO Exception Caught.”; 
echo 'Error with the database:<br/>"; 
echo 'SQL Query: '.$query; 
echo '<pre>' 

echo "Error: " . $e->getMessage(). "<br/>"; 
echo "Code: " . $e->getCode(). "<br/>"; 
echo "File: " . $e->getFile(). "<br/>"; 
echo "Line: " . $e->getLine(). "<br/>"; 
echo "Trace: " . $e->getTraceAsString(). "<br/>"; 
echo '</pre>"; 


?> 


其 运行 结果 如 图 21.12 所 示 。 


阳 辕 村 的 


图 21.12 ”通过 存储 过 程 完成 用 户 的 注册 
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21.9 实 战 
区 视频 讲解 : 光盘 \TM\Video\ 第 21 章 \ 实 战 .exe 


21.9.1 通过 PDO 更 新 数据 库 中 数据 


例 21.11 在 利用 MySQL 数据 库 函 数 更 新 数据 时 ,需要 将 要 更 改 的 数据 信息 id 拼接 到 地 址 栏 的 参数 中 。 
在 PDO 中 操作 数据 更 新 也 不 例外 。 虽 然 利 用 PDO 操作 数据 的 方法 是 不 同 的 ， 但 是 原理 是 相同 的 。《〈 实 例 
位 置 : 光盘 \TMNInstance\21\21.11) 

首先 , 创建 脚本 文件 mdex.php, 编写 表格 , 设置 PDO 抽象 层 的 相关 参数 拼接 $dsn 变量 。 其 次 , 判断 “ 修 
改 ” 按 钮 是 否 被 单 击 ， 当 “修改 ”按钮 被单 击 时 ， 获 取 地 址 栏 传 递 的 参数 并 在 表格 最 下 方 动态 创建 form 表 
单 ， 其 中 包括 3 个 文本 框 和 一 个 提交 按钮 。 文 本 框 中 的 value 属性 值 由 PHP 提供 数据 并 设置 id 的 文本 框 ， 
将 其 设置 为 只 读 属性 不 可 更 改 。 最 后 ， 当 单 击 “确定 ” 按 钮 时 ， 通 过 PDO 的 exec0 方 法 执行 更 新 操作 。 其 
代码 如 下 : 


<?php 
try{ 
$pdo=new PDO($dsn, $user,$pwd); /实例 化 对 象 
$sql="select * from tb_pdo"; /查询 SQL 语句 
$result=$pdo->query($sql); 1/ 返回 结 果 集 
foreach($result as $valueX{ 
?> 
<tr> 
<td class="one"><?php echo $value[0];?></td> /| 循环 输出 数据 
<td class="one"><?php echo $value[1];?></td> 
<td class="one"><?php echo $value[2];?></td> 
<td class="one"><?php echo $value[3];?></td> 
<td class="one"><a href="index.php?id=<?php echo $value[0];?>"> 修 改 </a></td> 1/ 拼接 地址 栏 参 数 
</tr> 
<form action="method="post"> 
<?php 
} 
if(isset($_GETIid])X // 当 页 面 中 存在 id 变量 时 
$sql1="select * from tb_pdo where id='$_GETI[id]"; /拼接 SQL 语句 
Sresul=$pdo->query($sql1); /| 返回 结果 集 
foreach ($resul as $value}{ /| 循环 输出 结果 
?> 
<tr> 
<td class="one"><input readonly type='text name='id' value='<?php echo $value[id];?>'></td> 
<td class="one"><input type='text name='user value='<?php echo $value[1];?>'></td> 
<td class="one"><input type='password name='pwd' value='<?php echo $value[2];?>'></td> 
<td class="one"><input type='text name='date' value='<?php echo $value[3];?>'></td> 
<td class="one"><input class="two" type='submit name='sub1' value=" 确 定 '></td> 
</tr> 
<?php 
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和 
} 
?> 
</form> 
<?php 
i$_POST[sub1]==" 确 定 "X{ // 当 页 面 中 存在 参数 sub1 时 
$sql2="update tb_pdo set username='$_POST[user]',userpwd='$_POST[pwd]',date='$_POST 
[date] where id='$_POST[idl"; /拼接 SQL 语句 
$resu=$pdo->exec($sql2); /执行 更 新 操作 
if($resu==1){ 
echo "<script>alert(' 更 新 数据 成 功 ');window.location.href='index.php'</script>"; 
} 
》 
} catch (Exception $e){ 
echo "ERROR!!!".$e->getMessage()."<br>"; 
} 
?> 


其 运行 结果 如 图 21.13 和 图 21.14 所 示 。 


加 及 村 PE CE] La 
or raofe 


， 1011-0t-09 下 区 
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21.13 更 新 数据 图 21.14 更 新 数据 成 功 


21.9.2 ”明日 书店 会 员 注册 


例 21.12 注册 系统 在 本 书 的 很 多 节 中 都 有 体现 ， 本 次 所 讲解 的 是 利用 PDO 抽象 层 知识 实现 的 会 员 注 
册 系 统 。 其 核心 思想 是 拼接 插入 的 SQL 代码 ， 将 文本 框 的 相关 信息 动态 的 添加 到 数据 表 中 。 实例 位 置 ; 
光盘 \TMNVInstance\21\21.12) 
首先 ， 创 建 脚本 文件 index.php。 在 脚本 文件 中 编写 表单 ， 设 置 一 个 “注册 ”按钮 和 一 个 “ 重 置 ”按钮 。 
其 次 ， 当 单 击 “ 注 册 ” 按 钮 时 ， 首 先 通过 PDO 连接 MySQL 数据 库 ， 实 例 化 PDO 对 象 并 利用 对 象 句柄 调用 
exec0 方 法 ， 向 数据 库 中 添加 数据 。 其 代码 如 下 : 
<?php 
$dbms='mysql': /定义 PDO 的 相关 参数 
$host="localhost’; 
$user='root'; 
$pwd="111", 
$dbName='db_database21"'; 
$dsn="$dbms:host=$host;:dbname=$dbName"; 


ifisset($_POST[sub])){ // 当 页 面 中 存在 sub 变量 时 
try{ 
S$pdo=new PDO($dsn,$user,$pwd); /实例 化 对 象 
$sql="insert into tb_zc values(",'$_POST[text],'$_POSTIpwd]','$_POST[qq],'$_POST[Imail] ,now())"; 
/拼接 SQL 语句 
$result=$pdo->exec($sql): /返回 结果 集 
if($result==1){ // 输 出 提示 


echo "<script>alert(' 注 册 成 功 ");</script>"; 
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} 
} catch (Exception $e) { 
echo "ERROR".$e->getMessage()."<br>"; // 输 出 异常 
} 
} 


> 


运行 效果 如 图 21.15 所 示 。 
21.9.3 ”添加 留言 信息 


例 21.13 ”添加 留言 信息 是 非常 普通 和 常见 的 模块 操作 ， 其 实 从 原理 上 说 与 注册 登录 操作 几乎 一 样 ， 笔 
者 意 在 告知 读者 PDO 操作 常见 的 几 种 模式 ， 使 用 读者 熟 能 生 巧 。 (实例 位 置 : 光盘 \TMNInstance\2l\21.13) 

本 应 用 与 注册 登录 模块 原理 相同 ， 这 里 就 不 在 阐述 操作 步骤 了 。 本 应 用 的 核心 代码 如 下 : 
<?php 

$dbms='mysql'; /定义 PDO 的 相关 参数 

$user='root'; 

$pwd='111"; 

$host='localhost'; 

$dbName='db_database21'; 

$dsn="$dbms:host=$host;:dbname=$dbName"; 


if(isset($_POST[sub])X{ // 当 页 面 中 存在 sub 变量 时 

try{ // 利 用 try…catch 捕获 异常 
$pdo = new PDO($dsn, $user,$pwd); // 实 例 化 对 象 
$pdo->query("SET NAMES utf8"); // 设 置 页 面 的 编码 集 风格 
$sql="insert into tb_fb values(",'$_POSTI[title]','$_POST[content]',now())"; /定义 SQL 语句 
$rs=$pdo->exec($sql); /执行 插入 操作 
if$rs=="1")f 

echo "<b> 新 闻 发 布 成 功 </b>"; 

L 

} catch (Exception $e) { 


echo "ERROR".$e->getMessage()."<br>"; 
bp 


了 


运行 效果 如 图 21.16 所 示 。 
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21.15 ”会员 注册 系统 图 21.16 添加 留言 信息 
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21.9.4 查询 留言 内 容 


例 21.14 本 应 用 中 的 查询 留言 内 容 模块 是 一 个 典型 的 站 内 搜索 关键 字 描 红 的 基础 操作 。 究 其 根源 ， 其 
实 PDO 的 操作 并 不 复杂 ， 它 只 是 提供 了 一 个 连接 多 种 数据 库 的 一 个 统一 的 接口 。 至 于 具体 的 操作 还 是 来 自 
于 SQL 语句 本 身 。〔 实 例 位 置 ， 光盘 \TM\Instance\21\21.14) 

操作 步骤 如 下 。 

(1) 创建 脚本 文件 index.php。 定 义 form 表单 ， 设 置 一 个 文本 框 和 一 个 “查询 ”按钮 。 

(2) 当 “ 查 询 ” 按 钮 被 单 击 时 ， 首先, 判断 文本 框 内 容 是 否 为 空 , 其 次 , 使 用 PDO 抽象 层 连接 MySQL 
数据 库 ， 并 在 try…catch 内 部 利用 PDO 对 象 句柄 调用 query0 函 数 执行 查询 操作 ， 最 后 ， 输 出 查询 结果 。 其 
核心 代码 如 下 : 

<?php 
if(isset($_POSTIsub])X{ // 判 断 页 面 是 否 存在 sub 变量 
if($_POSTI[text]==" || $_POSTI[text]==" 输 入 查询 内 容 "){ // 判 断 文 本 框 内 容 是 否 为 空 
echo "文本 框 内容 不 能 为 空 "; 
}else{ 
$dbms="mysql"; /定义 PDO 的 相关 参数 
$user="root"; 
Spwd="111"; 
$host="|localhost"; 
$dbName="db_database21"; 
$dsn="$dbms:host=$host:dbname=$dbName"; 
try{ /try…catch 捕获 异常 
$pdo = new PDO($dsn,$user,$pwd); /实例 化 对 象 
$sql="select * from tb_fb where title like '%".$_POST[text]."%"; /拼接 SQL 语句 
$pdo->query("SET NAMES utf8 ); 1/ 设置 页 面 数据 的 编码 风格 
S$rs=$pdo->query($sql); 
foreach($rs as $value}{ /将 数据 循环 输出 


Ta 


<td class="f"><?php echo $rs [0];?></td> 
<td class="f"><?php echo $rs [1];?></td> 
<td class="f"><?php echo $rs [2];?></td> 
<td class="f"><?php echo $rs [3];?></td> 


<?php 


} 
} catch (Exception $e) { 

echo "ERROR".$e->getMessage()."<br>"; 
} 


中 
} 
?> 
运行 效果 如 图 21.17 所 示 。 
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2.10 水 结 


本 章 重点 介绍 数据 库 抽象 层 一 一 PDO， 从 它 的 概述 、 特 点 和 安装 开始 讲解 ， 到 它 的 实际 应 用 ， 包 括 如 
何 连 接 不 同 的 数据 库 、 如 何 执行 SQL 语句 、 如 何 获取 结果 集 ， 以 及 错误 处 理 ， 再 到 它 的 高 级 应 用 事务 
和 存储 过 程 都 进行 了 详细 讲解 ， 并 且 都 配 有 相应 的 实例 。 通 过 本 章 的 学 习 ， 相 信 读 者 能 够 掌握 PDO 技术 的 
应 用 。 


21.11 学 习 成 果 检 验 


1. 通过 PDO 向 数据 库 中 添加 数据 。 (实例 位 置 : 光盘 \TM\InstanceW21W21.15) 
2. 通过 PDO 浏览 数据 库 中 数据 。 (答案 位 置 : 光盘 \TM\Instance21\21.16) 
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综合 实例 (四 ) 一 一 BCTY365 网 上 社区 


( 铝 ! 视频 讲解 ， 138 分 钟 ) 


所 谓 网 上 社区 是 指 包 括 BBS 论坛 、 聊 天 室 、 情 客 等 形式 在 内 的 网 
上 交流 空间 ， 同 一 主题 的 网 上 社区 集中 了 具有 相同 兴趣 的 访问 者 ， 由 
于 有 众多 用 户 的 参与 ， 因 此 具备 交流 的 功能 。 

网 上 社区 有 各 种 不 同 的 表现 形式 和 规模 ， 由 个 人 创办 的 社区 ， 功 
能 和 界面 追求 时 尚 、 个 性 突出 大 型 的 商业 性 质 社区 ， 以 盈利 为 目的 ， 
分 类 多 元 化 ， 适 合 不 同类 型 的 网 民 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

H 了 解 网 上 社区 开发 的 基本 过 程 

H 了 解 如 何 做 需求 分 析 和 系统 设计 

MH 了 解 如 何 设计 和 实现 下 载 功能 

MH 了 解 在 线 论坛 功能 的 实现 方法 

MH 了 解 注册 模块 设计 

MH 了解 技术 支持 模块 设计 

MH 了 解 后 台 首 页 设计 

Mp 了 解 在 线 支 付 技 术 
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22.1 BCTY365 网 上 社区 概述 


本 章 开 发 的 BCTY365 网 上 社区 主要 面向 程序 开发 人 员 ， 集 论坛 、 留 言 板 、 软 件 下 载 、 升 级 下 载 、 技 术 


支持 和 在 线 购物 等 功能 于 一 身 ， 既 是 一 个 程序 开发 者 交流 的 平台 ， 也 是 一 个 网 络 营 销 的 场所 。 


22.1.1 系统 功能 结构 流程 


BCTY365 网 上 社区 系统 主要 分 为 前 台 和 后 台 两 部 分 ， 为 了 使 读者 能 够 更 清楚 地 了 解 网 站 的 结构 ， 


分 别 给 出 BCTY365 网 上 社区 前 台 和 后 台 功 能 模块 结构 图 。 
BCTY365 网 上 社区 前 台 管理 系统 的 功能 设计 如 图 22.1 所 示 。 
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图 22.1 网 上 社区 前 台 功 能 模块 结构 图 
BCTY365 网 上 社区 后 台 管理 系统 的 功能 设计 如 图 22.2 所 示 。 
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图 22.2 网 上 社区 后 台 功 能 模块 结构 图 
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22.1.2 ”系统 预览 


BCTY365 网 上 社区 系统 由 多 个 程序 页 面 组 成 ,下 面 给 出 几 个 典型 页 面 , 其 他 页 面 参见 光盘 中 的 源 程 序 。 
前 台 首页 如 图 22.3 所 示 ， 该 页 面 用 于 展示 本 系统 的 功能 模块 ， 突 出 企业 的 形象 ， 推 广 企业 的 软件 产品 。 
后 台 首 页 如 图 22.4 所 示 ， 该 页 面 用 于 实现 对 编程 词典 、 技 术 支持 、 软 件 升级 、 软 件 试用 等 内 容 的 管理 。 


| a 本 5 
rr 


区 3 rp 


— -上 er A 
22.3 前 台 首页 图 22.4 后 台 首 页 


在 线 订 购 模块 的 页 面 效 果 如 图 22.5 所 示 ， 该 页 面 主要 用 于 展示 本 企业 在 线 推出 的 软件 产品 ， 实 现 对 产 
品 的 在 线 购买 功能 。 软 件 下 载 模块 的 页 面 效果 如 图 22.6 所 示 ， 该 页 面 主要 用 于 展示 本 企业 提供 的 免费 软件 ， 
并 且 提供 下 载 链接 。 


旧 8CTY365 社 区 网 ， 找 你 想 要 的 
Er 


mre 


图 22.5 在 线 订 购 图 22.6 软件 下 载 
社区 论坛 模块 的 页 面 效 果 如 图 22.7 所 示 ， 该 页 面 主要 用 于 展示 论坛 中 的 各 大 版 块 ， 并 且 提 供 超 链接 跳 
转 到 对 应 的 版 块 。 后 台 的 登录 页 面 效果 如 图 22.8 所 示 ， 该 页 面 主要 实现 后 台 管 理 员 的 登录 。 
7 pm 


图 22.7 社区 论坛 图 22.8 后 台 登 录 
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22.2 ”数据 库 设 计 


鳃 q 视频 讲解 : 光盘 \TM\Video\ 第 22 章 \ 数 据 库 设计 .exe 

开发 一 个 功能 完善 的 网 上 社区 离 不 开 数 据 库 的 支持 ， 只 有 拥有 了 强大 的 数据 库 ， 网 上 社区 才能 够 存储 
大 量 的 数据 信息 ， 实 现 更 多 、 更 好 的 功能 来 吸引 更 多 的 社区 成 员 。 本 节 将 对 BCTY365 网 上 社区 数据 库 的 设 
计 进 行 详细 介绍 。 


22.2.1 数据库 概要 说 明 


在 BCTY365 网 上 社区 系统 中 应 用 的 是 db_bety365 数据 和 
库 ， 其 中 涉及 18 个 数据 表 ， 数 据 表 的 名 称 和 功能 如 图 22.9 。。 So pone mas 


所 示 。 hbbs WYIBAM 0b2312_chineae_ 局 。 论坛 发 帖 语 息 机 
有 b_bccd MYISAM gb2312_chinese_d 。 请 棍 词 内 舍 忆 来 
如 cy MyWSAM 952312_chmese_cl 。 荐 市 信 四 素 
念 设 j er rr 
22.2.2 ”数据 库 概念 设 让 
有 epty WyISAM 962312_chinese_H 论坛 回 企 信息 杰 
Si MYISAM go2312_chineso_ol 。 软 促 升 组 下 就 信息 表 
了 Sof WYBAM 0b2312_chineae_ 可。 软件 下 就 情 自 各 
根据 上 述 各 节 对 BCTY365 网 上 社区 系统 做 的 需求 分 析 和 RD A 


pe WISAM 952312_chinese_cl 社区 大 志江 信息 可 


系统 设计 ， 规 划 出 BCTY365 网 上 社区 的 实体 关系 E-R 图 。 其 和 pe bg。 Wh 902512 chineoe 9 。 认 大 关 和 机 
中 包括 注册 用 户 实体 、 发 帖 信息 实体 、 回 帖 信息 实体 、 订 单 信 EEE EEE RE 
息 实 体 、 编 程 词典 信息 实体 ， 以 及 一 些 辅助 实体 , 用 于 对 上 述 Cr 
实体 进行 补充 。 由 于 涉及 的 实体 较 多 , 这 里 只 对 注册 用 户 实体 、 Wenenodl WBN DZhnean PRRR 
发 帖 信息 实体 和 订单 信息 实体 的 E-R 图 进行 介绍 。 22.9 db_bcty365 数据 库 中 使 用 的 数据 表 
1. 注册 用 户 实体 
注册 用 户 实体 用 于 存储 用 户 注册 信息 ， 包 括 编 号 、 用 户 名 、 真 实 姓名 、 密 码 、 邮 箱 、 性 别 、 电 话 、QQ 
号 码 、 家 庭 地 址 、 访 问 次 数 、 注 册 时 间 、 最 后 一 次 登录 时 间 、IP 地 址 、 邮 政 编码 、 用 户 类 型 、 密 码 提示 问 
题 、 密 码 答案 、 真 实 密码 、 表 情 图 和 发 帖 次 数 属性 。 注 册 用 户 实体 的 E-R 图 如 图 22.10 所 示 。 
2. 发 帖 信息 实体 
发 帖 信息 实体 用 于 存储 登录 本 社区 的 会 员 在 论坛 中 发 布 帖子 的 相关 信息 ， 包 括 编 号 、 用 户 名 ID、 帖 子 
类 型 、 帖 子 标题 、 帖 子 内 容 、 发 帖 时 间 、 最 后 回复 时 间 、 表 情 图 、 访 问 次 数 、 是 否 顶 帖 和 上 传 图 片 属性 。 
发 帖 信息 实体 的 E-R 图 如 图 22.11 所 示 。 


22.10 注册 用 户 实体 E-R 图 22.11 发帖 信息 实体 的 E-R 图 
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3. 订单 信息 实体 

订单 信息 实体 存储 用 户 在 线 购买 时 填写 的 订单 信息 ， 包 括 编号 、 用 户 名 、 性 别 、 家 庭 地 址 、 邮 政 编码 、 
QQ 号 码 、 邮 箱 、 手 机 号 码 、 电 话 号 码 、 收 货 方式 、 邮 资 、 产 品 金 额 、 订 单 时 间 、 订 单 号 和 选择 城市 等 属性 。 
订单 信息 实体 的 E-R 图 如 图 22.12 所 示 。 


22.2.3 数据库 远 辑 设计 


本 项 目 中 创建 数据 库 和 数据 表 使 用 的 是 phpMyAdmin 图 形 化 管理 工具 。 下 面 将 介绍 数据 库 和 数据 表 的 
创建 方法 ， 以 及 在 创建 过 程 中 需要 注意 的 一 些 问 题 。 

1. 数据 库 的 创建 

打开 phpMyAdmin 图 形 化 管理 工具 的 主页 ， 首 先 在 文本 框 中 输入 要 创建 的 数据 库 的 名 称 〈 如 
db_bcty365), 然后 在 下 拉 列 表 框 中 选择 要 使 用 的 字符 编码 格式 , 这 里 使 用 的 是 gb2312_chinese ci, 如 图 22.13 
所 示 。 最 后 单 击 “ 创 建 ”按钮 ， 数 据 库 创 建成 功 。 


localhost phpMyAdmin - 2.9.0.2 


现 服务 器 lccalhos 人 re 国 :| 四 文 -Chinese smplfed 国 


全 pnpMyAomin 宫 方 网 站 
i 

i 

NT |[ 选 择 编 权 格式 


» [ChangeLogl [CVS] [Usts] 
Bport hpMyAdmiIn 
亚 登 出 田 phpMue 


22.12 订单 信息 实体 E-R 22.13 phpMyAdmin 管理 界面 


让 


[a 技巧 创建 数据 库 的 过 程 中 ,尽量 使 用 与 程序 内 容 贴 切 的 英文 字符 进行 命名 ,有 助 于 对 数据 库 的 理解 。 
如 果 使 用 AppServ 配置 PHP 开发 环境 那么 在 创建 数据 库 时 不 需要 指定 编码 的 格式 ， 默 认 值 为 
gb2312_chinese_ci; 如 果 自 行 配置 开发 环境 ， 那 么 就 要 指定 编码 格式 为 gb2312_chinese_ci， 否 则 创建 数 
据 库 的 编码 格式 为 latinl_swedish_ ci， 将 导致 数据 库 中 数据 出 现 乱 码 。 


2. 创建 数据 表 

在 成 功 创建 数据 库 后 ， 接 下 来 就 是 创建 数据 表 ， 这 里 以 tb_bb 编程 词典 版 本 信息 表 为 例 ， 讲 解 如 何 创建 
数据 表 ， 以 及 在 创建 数据 表 的 过 程 中 需要 注意 哪些 问题 。 这 里 创建 一 个 名 字 为 tb_bb 的 数据 表 ， 包 括 3 个 字 
段 ， 如 图 22.14 所 示 。 

单 击 “ 执 行 ” 按 钮 后 ， 进 入 到 如 图 22.15 所 示 的 添加 字段 信息 的 页 面 中 ， 在 此 处 对 字段 进行 详细 设置 ， 
包括 字段 名 、 数 据 类 型 、 长 度 / 值 、 属 性 、 默 认 值 、 人 额外、 主键 和 索引 等 。 


服务 器 : localhost } 避 效 据 库 : db_bcty365 四 老 :tb_bb 
为 在 数据 库 db_bcty365 中 创建 一 个 新 表 
各 Number offiel(® B_) 
输入 新 表 名 称 输入 新 表 字 段 个 数 执行 


Sy 二 更] 网 天 其 加 | 芭 | 二 右 | 芭 训 丰 ER 
加 cy 司 后 nornu 一 习 
22.14 创建 也 bb 数据 表 图 22.15 ”添加 数据 表 中 的 字段 信息 


[ 副 攻 2 
指定 编码 格式 | 


@ 
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二 


La 技巧 创建 数据 库 中 的 数据 表 时 , 字段 名 的 设计 要 尽量 与 数据 表 的 内 容 相 符合 , 这 样 有 助 于 程序 后 期 
维护 和 修改 工作 的 进行 ， 能 够 直观 地 看 出 数据 表 的 作用 .。 

如 果 使 用 AppServ 配置 PHP 开发 环境 ,那么 在 创建 数据 表 时 不 需要 指定 数据 表 类 型 和 编码 格式 ; 如 
果 使 用 自行 配置 的 PHP 开发 环境 ， 那 么 就 要 指定 数据 表 的 类 型 为 MYISAM， 字 符 的 编码 格式 为 
gb2312_chinese_ ci， 和 否则 创建 的 数据 表 类 型 为 IhnoDB， 而 编码 格式 为 latinl swedish_ ci， 这 将 导致 该 数 
据 表 中 的 数据 复制 到 其 他 计算 机 上 不 可 用 ， 并 且 数 据 表 中 的 数据 出 现 乱 码 。 
在 创建 数据 表 的 过 程 中 ， 一 定 要 为 数据 表 指 定 一 个 主键 ， 它 是 数据 表 唯 一 的 标识 。 


掌握 数据 表 的 创建 方法 后 ， 就 可 以 自行 创建 数据 表 。 由 于 本 项 目 中 涉及 的 数据 表 有 17 个 之 多 ， 这 里 不 
能 对 每 个 数据 表 的 功能 设计 进行 一 一 介绍 ， 所 以 只 给 出 几 个 重要 的 数据 表 的 设计 效果 图 供 广 大 读者 参考 ， 
其 他 数据 表 请 参见 本 书 附带 的 光盘 。 
(1) tb_user (注册 用 户 信息 表 ) 
注册 用 户 信息 表 主 要 用 于 存储 本 社区 中 会 员 的 个 人 信息 。 该 数据 表 的 结构 如 图 22.16 所 示 。 


国 服务 器 localhost ， 局 数据 床 ; db_bety365 ， 辐 表 :tb_user 
FR Ba 要 时 硬性 Nul 器 认 


下 开明 
u Pe 百 rn_inerernenl 自动 电导 
eorme vorenanso) e232 ennose el ur 证 和 用 P 各 
aename varshens0) -2312 sminese ol 是 NUL FRE 
md rzhanGGO。 由 2912_rhinese_o 时 MUL De 
washan(so) 用 2313_rhnesw_5 NUL 和 
sox yorehonD W2212 shinoeo_ ol 时 NUL Ba] 

加 rarhar2O 。 号 2312_rinese_ol 是 AULL 贡 系 电 活 
uch) 2942 chinese al 是 NULL Co 
agrees varenart1o0) P2312 ennose_ el ur [0 
togntimes nia 可 Wy 
otime ome 百 [| 
astogmme sate 百 是 后 一 次 和 
中 yorehon20) 。 了 2312_zhnaca_o 再 JI 

内 meat20 e2312 chinese ol NU 
Here mi 至 必 P 赤 型 
esten 。。 warsnarz0b) 号 2312_eninece 再 Ra 检 示 月 轨 
aswer Yareher200) ob2912 shinoseo 和 
amd varharCrn) ob2317_chinese 9 百 He 
ne Yarnar5Q。 用 2313_rnnace_o 至 地 四 
pubtimes nt) 是 2 


22.16 注册 用 户 信息 表 结 构 


(2) tb_reply《〈 论 坛 回帖 信息 表 ) 
论坛 回帖 信息 表 主 要 用 于 存储 登录 会 员 在 本 社区 中 回复 帖子 的 信息 。 该 数据 表 的 结构 如 图 22.17 所 示 。 
(3) tb_bccd〈 编 程 词典 信息 表 ) 
编程 词典 信息 表 主 要 用 于 存储 本 社区 在 线 订 购 模块 中 出 售 的 编程 词典 的 基本 信息 。 该 数据 表 的 结构 如 
图 22.18 所 示 。 
DLR Chi bety355 》 国 素 :tb_bocd 


野性 Nul 加 认 阁 外 说 明 

hy Em 再 Atloneremen Nan 

字 民 a 里 竹 Mul 至 认 如 曙 ccdmame 。 vareanCom oh2312_chinese_rl 至 亲人 同 奥 和 和 
a nko) 和 要 aut_incramert a4 访 二 DD ‘Owner arrnal0g gb2312_chinese_cl 可 开发 天 
userid in 再 0 出 下 PD Whe Varsnet(S0) ob2312_chinese_ ci 至 版 相关 型 
bbsid In) 再 0 长 有 由 子 世 I Econtent madiomted ob2312_chinese_ci 于 铺 得 司 奥 障 介 
me aharom gb2312_eminase_el 是 ”MOL C53 Samepart modiurtot ch2312.chinese_ ri 否 计件 共同 
comtork 。 modumiot gb2312 nmoco el 是 ML ap ae aiotme 理 有 
createtime datetme 是 ALL shia imageadiress varnatt100) gb2312_chinesa_ ci 再 人 地 引 
ma te 是 MUL ge ba me 是 WL om 
photo ‘varcharid0) ob2312_chinese_c] 是 WUEL EE ie Mcat 是 WM 省 格 

图 22.17 论坛 回帖 信息 表 结 构 图 22.18 编程 词典 信息 表 结 构 


@ 
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22.3 前 台 首 页 设计 


村 | 视频 讲解 : 光盘 \TM\Video\ 第 22 章 \ 前 台 首 页 设计 .exe 

当今 时 代 ， 很 多 人 都 十 分 重视 事物 的 第 一 印象 ， 在 网 络 中 更 是 如 此 ， 网 站 给 人 的 第 一 印象 如 果 不 好 ， 
那么 就 会 有 很 多 人 不 去 浏览 该 网 站 ， 无 论 网 站 的 内 容 是 否 丰富 。 可 以 说 网 站 首页 设计 的 成 功 与 否 直接 影响 
着 整个 网 站 的 发 展 。 


22.31 


前 台 首 页 概述 


网 站 首页 是 整个 网 站 的 脸面 ， 既 要 突出 企业 的 形象 ， 又 要 展示 出 网 站 强大 的 功能 。 如 果 网 站 首页 设计 
得 非常 成 功 ， 那 无 疑 为 整个 网 站 的 成 功 增添 了 一 个 硅 码 。BCTY365 网 上 社区 首页 的 设计 以 企业 的 品牌 形象 


为 基础 ， 


因 因 办 办 办 办 罗 


全 力 打造 网 站 的 整体 功能 ， 重 点 推出 企业 的 软件 产品 。 具 体内 容 如 下 所 示 。 

网 站 菜单 导航 : 包括 首页 、 技 术 支持 、 在 线 订 购 、 社 区 论坛 、 软 件 下 载 、 升 级 下 载 、 购 买 须知 和 
联系 我 们 。 

用 户 注 册 和 登录 模块 : 实现 用 户 注册 、 会 员 登录 、 找 回 密码 和 修改 密码 的 功能 。 

网 站 公告 ， 主 要 用 于 发 布 社区 中 的 一 些 新 消息 和 重大 事件 。 

编程 词典 模块 ， 推 广 企业 的 软件 产品 。 

软件 下 载 模块 ， 展 示 企 业 提 供 的 试用 版 和 免费 的 软件 产品 。 

常见 问题 模块 ， 列 举 出 编程 中 常见 问题 及 其 解决 方案 。 

社区 论坛 模块 : 浏览 社区 论坛 中 的 部 分 帖子 。 

升级 下 载 模块 : 提供 一 些 软件 的 升级 版 本 下 载 。 


上 述 就 是 BCTY365 网 上 社区 首页 体现 的 内 容 ， 为 了 更 加 直观 地 了 解 网 上 社区 首页 的 设计 ， 这 里 先 预览 


一 下 社 


区 首页 ， 该 首页 在 本 书 光盘 中 的 路 径 为 光盘 \TM\22\index.php， 如 图 22.19 所 示 。 


TT ps rr 


图 22.19 BCTY365 网 上 社区 首页 
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BCTY365 网 上 社区 首页 的 设计 看 上 去 很 复杂 ， 由 很 多 版 块 组 成 ， 但 实现 的 过 程 非常 简单 。 总 体 架 构 
使 用 一 个 2 行 3 列表 格 和 一 个 3 行 3 列表 格 ， 将 其 分 割 成 不 同 的 版 块 ， 然 后 使 用 脚本 语句 从 数据 库 中 读 取 
数据 ， 最 后 将 数据 循环 输出 到 页 面 中 ， 其 中 网 站 的 头 尾 文件 使 用 include 包含 语句 调用 。 首 页 的 框架 结构 如 
图 22.20 所 示 。 


钨 种 鹿 旨 次 荐 


22.20 ”网 站 首页 的 框架 结构 


22.3.2 ”公告 信息 的 滚动 输出 技术 


作为 网 站 首页 ， 不 一 定 要 具有 特殊 的 功能 ， 而 应 该 简洁 、 鲜 明 ， 以 突出 企业 形象 、 展 示 网 站 的 功能 为 
主 。 即 使 使 用 的 是 一 个 静态 页 面 ， 只 要 能 够 将 内 容 表达 全 面 、 完 整 ， 那 么 这 个 首页 设计 也 是 非常 成 功 的 。 

在 本 项 目 首页 的 设计 中 ， 应 用 到 一 个 文字 循环 滚动 的 技术 ， 通 过 该 技术 来 输出 社区 中 发 布 的 公告 信息 。 
该 技术 是 通过 JavaScript 脚本 和 <div> 标 签 来 共同 实现 的 ， 其 实现 的 原理 是 : 首先 创建 一 个 <div> 标 签 ， 然 后 
在 <div> 标 签 中 输出 公告 信息 ,最 后 通过 JavaScript 来 对 <div> 标 签 进行 操作 , 实现 <div> 标 签 中 内 容 的 滚动 输 
出 。 该 技术 的 实现 在 index.php 页 中 完成 。 其 中 使 用 的 JavaScript 脚本 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\22\index.php) 

<script language="JavaScript"> 


marqueesHeight=222; /定义 输出 标签 的 高 度 
stopscroll=false; /定义 stopscroll 的 默认 值 为 false 
with(marquees){ /| 编辑 marquees 标签 的 属性 
style.width=0; // 定 义 初始 宽 为 0 
style.height=marqueesHeight; /定义 marquees 标签 的 高 为 222 
style.overflowX="visible"; /定义 值 为 显示 
style.overflowY="hidden"; /定义 值 为 隐藏 
noWrap=true; 
onmouseover=new Function("stopscroll=true"); // 当 光标 经 过 时 执行 stopscroll=true 
onmouseout=new Function("stopscroll=false"); // 当 光标 离开 时 执行 stopscroll=false 


} 
// 创 建 一 个 新 的 div"templayer" 与 div"marquees" 进 行 连接 ， 实 现 不 间断 地 循环 输出 内 容 
document.write('<div id="templayer" style="position:absolute;z-index:1;visibility:hidden"></div>"); 
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preTop=0; currentTop=0; 
function init(){ 
templayer.innerHTML="™"; // 设 置 templayer 的 初始 值 为 空 
while(templayer.offsetHeight<marqueesHeight{ 。”“”// 判 断 当 templayer 的 高 度 小 于 marquees 的 高 度 时 
templayer.innerHTML+=marquees.innerHTML; // 将 templayer 的 值 赋 给 marquees 


} 
marquees.innerHTML=templayer.innerHTML+templayer.innerHTML; // 将 templayer 的 值 累加 
setlnterval("scrollup()",50); /间隔 50 毫秒 执行 一 次 scrollup() 函 数 
} 
function scrollup(X{ /实现 滚动 输出 
if(stopscroll==true) return; / 潮 断 如 果 stopscroll==true， 不 执行 循环 
preTop=marquees.scrollTop; 
marquees.scrollTop+=1; 
if(preTop==marquees.scrollTop)\{ 
marquees.scrollTop=templayer.offsetHeight-marqueesHeight; 
marquees.scrollTop+=1; 
) 
} 
</script> 


在 <div> 标 签 中 ， 主 要 是 输出 数据 库 中 存储 的 公告 信息 ， 并 且 对 输出 的 信息 进行 截取 和 蔡 换 ， 规 范 输出 
的 内 容 。<div> 标 签 中 的 程序 代码 如 下 : 
<div id=marquees > <!--> 创 建 一 个 div 标签 <!--> 
<table width="200" height="25" border="0" align="center” cellpadding="0" cellspacing="0"> 
<?php // 从 数据 库 中 读 取 公告 数据 
$sql=mysql_query("select id ,title,createtime from tb_tell order by createtime desc limit 0,10",$conn); 
$info=mysql_fetch_array($sql); 


if($info==false){ /判断 当 $info==false 时 执行 下 面 的 内 容 
sp 
<tr> 
<td height="25"><div align="center"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div></td> 
</tr> 
<?php  }else{ 
$i=1; // 定 义 变量 $i=1 
dof /执行 do…while 循环 语句 
?> 
<tr> 


<td height="25" style="padding:4"> 
<a href="tellinfo.php?id=<?php echo S$info['id'"];?>" class="a1"> 
<?php 
if($i==1){ // 当 $i==1 时 ， 将 输出 的 内 容 设置 为 红色 
echo "<font color=red>"; 


» 
echo $i.".&nbsp;"; 


echo unhtml(msubstr($infoftitle].0,50)); /应 用 自 定义 函数 对 输出 的 内 容 进行 控制 

if(strlen(S$infor'title])>50X{ // 当 输出 内 容 的 长 度 超过 50 个 字符 时 用 “...” 代 圭 
echo"..."; 

和 


echo "(".str_replace("-","/",$info[createtime]).")": /将 输出 的 公告 时 间 中 的 “-” 用 “/” 替 代 
if$i==1){ echo "</font>"; } 
9 </a> 
</td> 
</tr> 
<?php 
Si 


国 
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}while(S$info=mysql_fetch_array($sq))); /1do*…while 循环 语句 结束 
} 


?> 
</table> 
</div> 


在 首页 中 使 用 滚动 条 是 一 个 不 错 的 方法 ， 添 加 滚动 条 可 以 增加 网 页 的 动态 效果 ， 提 高 网 页 的 观赏 性 ， 
而 且 不 会 影响 网 页 的 浏览 速度 。 


22.3.3 ”前台 首页 的 实现 过 程 


开发 网 站 首页 主要 就 是 连接 数据 库 , 然后 从 数据 库 中 读 取 数据 , 最 后 应 用 循环 语句 将 数据 库 中 数据 输出 
到 前 台 页 面 。 由 于 使 用 的 代码 较 多 ， 而 且 多 数 都 是 重复 使 用 ， 所 以 这 里 只 给 出 首页 中 公告 发 布 模块 的 代码 。 

公告 发 布 模块 主要 实现 从 数据 库 中 读 取 公 告 数据 ， 将 数据 在 首页 中 滚动 输出 ， 并 且 对 公告 信息 的 长 度 
进行 控制 ， 保 证 内 容 的 整齐 、 规 范 。 详 细 代码 可 以 参考 本 书 光 盘 中 TM\22\bcty365\index.php 文件 。index.php 
文件 的 部 分 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\22\index.php) 


<?php include_once("top.php"); ?> // 获 取 头 部 文件 
…// 省 略 部 分 代码 
<?php 
// 读 取 数 据 库 中 公告 表 中 的 数据 
$sql=mysql_query("select id,title,createtime from tb_tell order by createtime desc limit 0,10",$conn); 
$info=mysql_fetch_array($sql); /执行 读 取 数 据 表 中 数据 的 语句 
if($info==falseX{ // 如 果 返 回 值 为 空 则 执行 下 面 的 语句 
了 
<tr> 
<td height="25"><div align="center"><a href="#" class="a4"> 本 站 暂 无 公告 发 布 ! </a></div></td> 
</tr> 
<?php 
}else{ // 如 果 返 回 值 不 为 空 则 执行 下 面 的 do…while 循环 语句 
$i=1; 
dof 
?> 
<tr> 
<td height="25" style="padding:4"><a href="tellinfo.php?id=<?php echo $infor'id];?>" class="a1"> 
<?php 
if($i==1X{ // 当 变量 $i=1 时 ， 输 出 的 内 容 以 红色 字体 显示 
echo "<font color=red>"; 


} 
echo $i.".&nbsp;"; 
@ echounhtml(msubstr($info[title’],0,50)); 
@ if(stren($info[title])>50X{ // 如 果 标题 长 度 大 于 50 个 字符 ， 则 以 省 略 号 代替 


echo .~ 


} 
© echo"(".str replace("","/",$infof createtime"]).")"; // 输 出 公告 发 布 的 时 间 ， 并 且 将 其 中 的 “/” 使 用 “-” 蔡 换 
if($i==1){ 
echo "</font>"; 
} 


</a> 
</td> 
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<?php 

Sir+; /变量 自 加 1 
}while($info=mysql_fetch_array($sql)); /ldo…while 循环 语句 结束 
加 

加 /省略 了 部 分 代码 

<?php 

include_once("bottom.php"); // 调 用 网 站 的 尾 文件 

> 


a 
NR, a unhtmlO 和 msubsttO 自 定义 函数 去 除 输出 字符 串 中 的 空格 和 控制 给 出 字符 事 的 长 度 。 
@ strlen0 函 数 用 于 获取 指定 字符 串 的 长 度 。 
上 @ str_replace() 函 数 用 于 实现 字符 串 的 替换 。 


22.4 注册 模块 设计 
雪 4 视频 讲解 :光盘 \IM\Video\ 第 22 章 \ 注 册 模块 设计 .exe 


22.4.1 注册 模块 概述 


为 了 更 好 地 与 广大 网 民 朋 友 进 行 交流 和 沟通 ，BCTY365 网 上 社区 系统 中 创建 了 一 个 会 员 注 册 模 块 。 通 
过 会 员 注册 模块 ， 可 以 有 效 地 对 用 户 信息 进行 采集 ， 并 将 合法 的 用 户 信息 保存 到 指定 的 数据 表 中 ， 实 现 与 
用 户 的 长 期 沟通 和 交流 。 既 然 设置 了 会 员 注册 模块 ， 那 么 在 系统 中 就 要 为 会 员 提供 一 些 特殊 的 权限 。 在 本 
系统 中 注册 为 会 员 可 以 拥有 如 下 权限 : 在 本 社区 的 论坛 中 发 布 和 回复 帖子 、 在 技术 支持 模块 中 发 表 留 言 、 
在 升级 下 载 模块 中 下 载 软件 升级 包 等 , 而 且 可 以 修改 和 找 回 密码 。 用 户 注册 模块 的 运行 结果 如 图 22.21 所 示 。 
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图 22.21 用 户 注册 模块 的 运行 结果 


22.4.2 ”通过 JavaScript 脚本 验证 表单 元 素 
在 会 员 注册 模块 中 ， 必 不 可 少 的 功能 就 是 要 对 用 户 输入 的 信息 进行 判断 ， 首 先 判 断 用 户 填写 的 注册 信 


® 
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息 中 哪些 是 必须 填写 的 、 哪 些 可 以 不 填写 ， 然 后 进一步 判断 输入 的 信息 是 否 合理 、 合 法 ， 如 判断 输入 的 邮 
编 格式 是 否 正确 、 邮箱 格式 是 否 正 确 等 。 对 表单 中 提交 数据 进行 判断 最 常用 的 办 法 就 是 使 用 JavaScript 脚本 ， 
也 可 以 使 用 正则 表达 式 。 下 面 讲解 在 本 模块 中 是 如 何 通过 JavaScript 脚本 实现 表单 提交 数据 验证 。 

操作 原理 是 : 在 form 表单 中 调用 onsubmit 事件 , 通过 该 事件 调用 指定 的 JavaScript 脚本 , 执行 shkinputO 
自 定义 函数 ， 实 现 对 表单 中 提交 数据 的 验证 。 在 JavaScript 脚本 中 ， 对 表单 中 提交 数据 进行 判断 ， 判 断 输 入 
的 内 容 是 否 为 室 、 内 容 的 格式 是 否 正确 ， 如 果 正 确 则 继续 执行 ， 否 则 将 弹出 提示 对 话 框 ， 并 将 鼠标 的 焦点 
指定 到 出 错 的 位 置 。 具 体 的 JavaScript 脚本 代码 如 下 : 

<script language="JavaScript" type="text/javascript"> 


function chkinput(form}{ /定义 一 个 函数 

ifform _tel.value==""){ /判断 tel 文 本 框 中 的 值 是 否 为 空 
alert(" 请 填写 联系 电话 "); // 如 果 为 空 则 输出 “请 填写 联系 电话 !” 
form.tel.select(); // 返 回 到 tel 文本 框 
return(false); 

ifform.emailLvalue=="" /| 判断 email 文本 框 的 值 是否 为 空 

alert(" 请 输入 E-mail 地 址 ""); // 如 果 为 空 则 输出 “请 输入 E-mail 地 址 !” 

form.email.select(); /| 返回 到 email 文本 框 


return(false); 


var i=form.email.value.indexOf("@"); 
var j=form.email.value.indexOf("."); 
// 进 一 步 判 断 邮 箱 的 格式 是 否 正确 ， 是 否 包 含 “@” 和 “.” 
if((i<O)IIGi-j>0)G<0)X 
alert(" 请 输入 正确 的 E-mail 地 址 ""); 


form.email.select(); /| 返回 到 email 文本 框 
return(false); 
} 
return(true); /提交 表单 
} 
</script> 


上 述 代码 中 ， 只 是 列举 了 JavaScript 中 的 部 分 内 容 ， 并 且 在 对 电话 号 码 进行 判断 时 ， 只 是 判断 其 是 否 为 
空 ， 没 有 进一步 判断 电话 号 码 的 格式 是 否 正确 。 如 果 想 要 更 加 准确 地 判断 电话 号 码 的 格式 是 否 正确 ， 可 以 
采用 下 面 的 方法 : 通过 正则 表达 式 的 preg_match0 函 数 ， 在 表单 提交 处 理 页 中 对 电话 号 码 进行 判断 。 

preg_match0) 函 数 的 语法 格式 如 下 : 

int preg_match ( string pattern, string subject [, array matches [, int flags]] ) 

Preg_match() 函 数 的 参数 说 明 如 表 22.1 所 示 。 


表 22.1 preg_match() 函 数 的 参数 说 明 


参数 说 明 


_pattem | 必 选 参数 。 需 要 匹配 的 正则 表达 式 


Subject 必 选 参数 。 输 入 的 字符 串 


es | 可 选 参 数 。 输 出 的 搜索 结果 的 数组 ， 如 Soutfo] 将 包含 与 整个 模式 匹配 的 结果 ，Sout[l] 将 包含 与 第 一 个 捕 
“| 获 的 括号 中 的 子 模式 所 匹配 的 结果 ， 依 次 类 推 


可 选 参 数 。 若 设 为 PREG_OFFSET_CAPTURE， 对 每 个 出 现 的 匹配 结果 也 同时 返回 其 附属 的 字符 串 偏 移 
量 ， 本 标记 自 PHP 4.3.0 起 可 用 


flags 


通过 preg_match0 函 数 判断 电话 号 码 的 格式 是 否 正确 的 方法 如 下 。 
首先 定义 一 个 用 于 判断 电话 号 码 格式 的 正则 表达 式 。 代 码 如 下 : 
人 Gd(3}-)Gdf8)SIACd(4-)Cdf7DSIACd(4-)Cd(8DS/ 


@ 
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正则 表达 式 的 功能 分 析 如 下 : 使 用 “^” 和 “$” 对 字符 串 进行 边界 的 限制 ， 对 区 号 从 字符 串 的 开始 进行 
匹配 ， 对 其 他 号 码 从 字符 串 的 末尾 开始 进行 匹配 ; 将 “0” 中 的 内 容 作 为 一 个 原子 使 用 ， 使 用 “\d” 来 匹配 
一 个 数字 ， 区 号 为 3 或 4 个 数字 ， 其 他 数字 为 7 或 8 个; 使 用 “ 丹 ” 来 对 前 字符 进行 重复 匹配 ; 使 用 “|” 
对 匹配 的 模式 进行 选择 ， 分 成 3 个 模式 。 
然后 将 该 正则 表达 式 应 用 到 preg_matchO 函 数 中 ， 对 表单 提交 的 电话 号 码 进行 判断 ， 如 果 正 确 则 继续 执 
行 ， 否 则 弹出 提示 信息 ， 并 返回 到 表单 提交 页 。 代 码 如 下 : 
<?ph 
‘Stel="0431-84978981": // 定 义 一 个 电话 号 码 的 变量 
if(preg_match("/*(\d{3}-)(\d{8})$I^(\d{4}-)(\d{7})SI*(d{4}-)(\d{8})$I^(d{11))$/", Stel, Scounts)\{ 
echo "您 输入 的 电话 号 码 格式 正确 !"; /输出 字符 串 
ls 
a Se "<script>alert(' 您 输入 的 电话 号 码 的 格式 不 正确 !");history.back()</script>"; 


?> 


22.4.3 ”注册 模块 的 实现 过 程 


“ 


注册 模块 的 实现 过 程 非常 简单 ， 首 先 阅 读 注 册 服务 条 款 ， 然 后 填写 注册 的 用 户 名 和 密码 ， 提 交 后 由 系 
统 判断 输入 的 用 户 名 是 否 被 占用 ， 如 果 未 被 占用 则 可 以 继续 注册 ， 填 写 详细 的 注册 信息 ， 将 数据 提交 到 表 
单 处 理 页 进行 处 理 ， 最 后 将 用 户 注册 的 信息 保存 到 指定 的 数据 表 中 。 用 户 注册 模块 的 实现 过 程 主要 由 3 个 
文件 完成 : registerphp 文件 用 于 输出 注册 服务 条 款 ， 以 及 填写 注册 的 用 户 名 和 密码 ， 并 且 判 断 注册 的 用 户 
名 是 否 被 占用 ; getuserinfo.php 文件 用 于 填写 详细 的 注册 信息 ， 并 且 在 表单 中 应 用 数字 验证 码 技术 ， 
savereginfo.php 文件 用 于 对 表单 中 提交 的 数据 进行 处 理 ， 将 数据 保存 到 指定 的 数据 表 中 。 

在 savereginfo.php 文件 中 ,首先 连接 数据 库 , 然后 获取 表单 中 提交 的 数据 ， 并 且 判 断 提交 的 用 户 名 是 否 
被 占用 ， 最 后 将 提交 的 数据 进行 处 理 ， 并 将 数据 保存 到 指定 的 数据 表 中 。 程 序 代 码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\22\saverreginfo.php) 


<?php session_start(); // 初 始 化 session 变量 

include_once("conn/conn.php"); // 连 接 数 据 库 

$usernc=trim($_POST[usernc]); /获取 注册 的 用 户 名 

// 判 断 指定 的 用 户 名 是 否 存在 

$sql=mysql_query("select usernc from tb_user where usernc=".$usemc."",$conn); 

Sinfo=mysql_fetch_array($sq)); /|/ 按 指定 条 件 检索 数据 信息 

if($info!=false}{ // 如 果 查 询 结果 不 为 空 ， 则 执行 以 下 操作 
echo "<script language='javascript>alert(' 对 不 起 ， 该 昵称 已 被 其 他 用 户 使 用 !");history.back();</script>"; 
exit; 

} 

$xym=trim($_POST[xym]); /去 除 变量 两 边 的 空格 

S$num=$_POST[num1]; /接收 变量 值 


ifstrval($xym)l=strval($num)){ 
echo "<script>alert(' 验 证 码 输入 错误 '");window.location.href='register.php';</script>"; 


exit;} 

// 对 表单 提交 的 数据 进行 处 理 

$truepwd=trim($_POST[pwd1]); // 获 取 真 实 密码 
S$pwd=md5($truepwd); // 获 取 加 密 密 码 
$truename=trim($_POST[truename]); // 获 取 真 实 姓名 
$email=trim($_POST[email]); /获取 邮箱 地 址 
$sex=$_POST['sex]; 1/ 获取 性 别 
Stel=trim($_POSTTtel]); /获取 电话 
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$yb=trim($_POST[yb]); // 获 取 邮 政 编码 
$qq=trim($_POST['qq]): /获取 QQ 
$address=trim($_POST[address]); /获取 地 址 
$question=trim($_POST[question]); /获取 提示 问题 
$answer=trim($_POST[answer]); // 获 取 问 题 答 案 
S$ip=getenv("REMOTE_ADDR"); /获取 客户 端的 IP 
Slogintimes=1; /指定 访问 次 数 
Sregtime=date("Y-m-j H:i:s"); 1/ 获取 时 间 
Slastlogintime=$regtime; 

Susertype=0; /指定 用 户 类 型 ， 默 认为 0 
$photo=$_POST["photo"]; 1/ 获取 头像 

// 将 表单 中 提交 的 数据 存储 到 数据 表 中 


if(mysql_query("insert into 
tb_user(usernc,truename,pwd,emailsex,tel,qq,address,logintimes,regtime,lastlogintime,ip,usertype,yb,question 
,answertruepwd,photo) 
values('$usernc','$truename','$pwd','$email','$sex','$tel','$qq','$address','$logintimes','$regtime','$lastlogintime',’ 
Sip'",'$usertype','$yb','$question','$answer','$truepwd','$photo')", $conn)}{ 

session_register("unc"); 

$_SESSION["unc"]=$usernc; 

echo "<script>alert(' 注 册 成 功 !");window.location.href='index.php';</script>"; 


jelsef // 如 果 添加 操作 失败 ， 则 给 出 提示 
echo "<script language='javascript>alert(" 对 不 起 ,注册 失败 !");history.back();</script>"; 
exit; // 退 出 

区 


22.5 技术 支持 模块 设计 


视频 讲解 : 光盘 \TM\Video\ 第 22 章 \ 技 术 支 持 模块 设计 .exe 
技术 支持 模块 主要 是 从 浏览 者 的 角度 进行 设计 ， 存 储 大 量 技术 问题 的 解决 方案 数据 ， 为 浏览 者 查阅 提 
供 方便 ， 而 且 设计 一 个 企业 与 客户 沟通 的 平台 ， 能 够 随时 了 解 客户 或 者 会 员 的 意见 和 需求 。 


22.5.1 技术 支持 模块 概述 


技术 支持 模块 主要 由 3 个 子 模块 组 成 ， 包 括 常 见 问题 模块 、 客 户 反 馈 模 块 和 联系 方式 模块 。 常 见 问题 
模块 主要 用 于 展示 编程 中 一 些 常见 问题 的 解决 方案 或 者 方法 ， 为 浏览 者 解决 编程 中 的 疑难 问题 提供 方便 ; 
客户 反馈 模块 主要 用 于 收集 和 获取 来 自 客户 的 需求 和 意见 ， 联系 方式 模块 主要 用 于 展示 企业 的 形象 和 具体 
的 联系 方式 。 


22.5.2 ”分 页 技术 


技术 支持 模块 中 ， 在 对 常见 问题 的 解决 方案 数据 进行 输出 时 ， 使 用 了 分 页 处 理 技术 ， 该 技术 的 设计 思 
路 是 : 从 数据 库 中 读 取 数据 ， 获 取 数据 总 量 ， 在 每 页 中 显示 20 条 数据 ， 根 据 数据 总 量 和 每 页 显示 的 条 数 对 
数据 进行 分 页 处 理 ， 计 算出 有 多 少 页 和 当前 显示 的 页 码 ， 实 现 首页 、 上 一 页 、 下 一 页 和 尾 页 之 间 的 页 面 跳 
转 。 具 体 的 设计 思路 可 以 参考 sjwtphp 文件 中 的 代码 注释 和 代码 贴 士 。cjwt.php 文件 的 程序 代码 如 下 : 


@ 
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(代码 位 置 ， 光盘 \TM\InstanceW22\cjwt.php) 


<?php 


$sql=mysql_query("select count(*) as total from tb_cjwt",$conn); 。”// 读 取 数 据 库 中 数据 
Sinfo=mysql_fetch_array($sql); 1/ 返回 数据 
@ Stotal=$infoftotal]; 


// 判 断 字段 total 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 
if($total==0X{ 
?> 
<tr> 
<td height="22" colspan="2"><div align="center"> 对 不 起 ， 暂 无 常见 问题 !</div></td> 
</tr> 
<?php 
Jelse{ // 如 果 不 为 空 ， 则 执行 下 面 的 内 容 
[2 if(!isset($_GET["page"]) || lis_numeric($_GET["page"])X // 判 断 $_GET 获取 的 page 的 值 是 否 存在 
© $page=1; /如果 不 存在 ， 则 设置 变量 的 值 为 1 
}else{ 
[4] $page=intval($_GET["page"]); /如 果 存 在 ， 则 获取 变量 $_GET 的 值 
} 
/设置 变量 $pagesize， 每 页 显示 的 数据 量 为 20 条 
© S$pagesize=20; 
if($total% S$pagesize==0X{ // 如 果 变 量 的 值 为 0 
© S$pagecount=intval($total/$pagesize); // 获 取 变 量 的 整数 值 
}else{ 
0 S$pagecount=ceil($total/$pagesize); // 如 果 不 为 0， 则 获取 实际 的 整数 值 
// 读 取 数 据 库 中 数据 ， 按 照 时 间 进 行 降 需 排 列 
$sql=mysql_query("select * from tb_cjwt order by createtime desc limit ".($page-1)*$pagesize.",$pagesize 
",$conn); 
while($info=mysql_fetch_array($sql)X{ 
?> 


… 1/ 省 略 部 分 代 


码 


<table width="600" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 


<tr> 


<td width="479"><div align="left*>&nbsp;&nbsp; 共 有 常见 问题 &nbsp;<?php echo $total;?>&nbsp; 条 &nbsp; 
每 页 显示 &nbsp;<?php echo $pagesize;?>&nbsp; 条 &nbsp; 第 &nbsp;<?php echo $page;?>&nbsp; 页 / 共 
&nbsp;<?php echo $pagecount;?>&nbsp; 页 </div></td> 
<td width="289"><div align="right"> 


<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=1" class="a1"> 首 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 


if($page>1) // 判 断 如 果 页 码 大 于 1 
echo $page-1; /| 输出 前 一 页 

else 
echo 1; 


?>" class="a1"> 上 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php 


if($page<$pagecount) // 如 果 页 码 小 于 总 页 数 
echo $page+1; 

else 
echo $pagecount' /输出 下 一 页 


?>" class="a1"> 下 一 页 </a>&nbsp; 
<a href="<?php echo $_SERVER["PHP_SELF"]?>?page=<?php echo S$pagecount;?>" 


class="a1"> 尾 页 </a>&nbsp;&nbsp;</div> 


</td> 
</tr> 
</table> 


全 
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4 
OC 培 归 @ Stotal: 为 数据 库 中 数据 总 的 记录 数 .。 
@ isset0: 检测 变量 是 否 已 经 设置 ; is_ numeric0: 检测 变量 是 否 为 数字 或 者 数字 字符 囊 。 
日 $page: 变量 为 页 码 中 的 第 几 页 。 
@ intval0: 获取 变量 的 整数 值 。 
@ Spagesize: 表示 在 每 页 中 显示 多 少 条 数据 。 
@ $pagecount: 表示 所 有 的 数据 可 以 分 成 多 少 页 。 
@ ceil0: 获取 变量 中 的 整数 值 ， 这 里 用 于 获取 页 码 的 整数 值 。 
@ 3_SERVER["PHP_SELF"]: 服务 器 变量 ， 这 里 用 于 获取 网 页 的 链接 地 址 。 


22.5.3 ”常见 问题 模块 的 实现 过 程 


常见 问题 子 模 块 实现 的 主要 功能 是 ， 展 示 出 数据 库 中 存储 的 有 关 编 程 中 遇 到 的 常见 问题 及 解决 方案 。 
其 运行 结果 如 图 22.22 所 示 。 
该 模块 由 两 个 文件 组 成 , 一 个 是 cjwt.php 文件 , 用 于 存储 创建 的 问题 数据 ; 另 一 个 是 lookcjwt.php 文件 ， 
用 于 输出 sjwtphp 文件 中 对 应 问题 的 详细 介绍 和 解决 方案 。 其 代码 如 下 : 
〈 代 码 位 置 ， 光盘 \TMNInstance\22\lookcjwtphp) 


<?php 
include_once("conn/conn.php"); /与 数据 库 建 立 连接 
include_once("function.php"); // 调 用 自 定义 函数 


// 读 取 tb_cjwt 表 中 的 数据 ， 条 件 为 jd="$_GET[id]" 
$sql=mysql_query("select * from tb_cjwt where id=".$_GET["id"]."",$conn); 
$info=mysql_fetch_array($sql); 

?> 


<table width="635" height="100" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 


<td width="94" height="30"><div align="center"><strong> 问 &nbsp;&nbsp; 题 : </strong></div></td> 
<td width="541"><?php echo unhtml($info["question"]); ?></td> // 输 出 问题 的 详细 内 容 
</tr> 
<tr> 
<td height="70"><div align="center"><strong> 解 &nbsp;&nbsp; 答 : </strong></div></td> 


<td height="70">&nbsp;<?php echo unhtml($info["answer]);?></td> /输出 问题 的 解决 方案 
</tr> 
</table> 


22.5.4 客户 反馈 模块 的 实现 过 程 


客户 反馈 子 模块 为 客户 提供 一 个 反馈 意见 和 提出 要 求 的 平台 ， 并 且 将 提交 的 信息 存储 到 数据 库 中 。 其 
运行 结果 如 图 22.23 所 示 。 
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图 22.22 常见 问题 模块 的 运行 结果 图 22.23 客户 反馈 模块 的 运行 结果 
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该 功能 只 对 本 网 站 中 的 会 员 开 通 ， 即 只 有 以 会 员 身 份 登录 的 用 户 才 具有 反馈 信息 的 权限 ， 其 中 在 对 提 
交 表单 的 细节 处 理 上 使 用 JavaScript 脚本 来 验证 表单 中 的 值 是 否 为 空 ， 而 且 还 使 用 数字 验证 码 技术 。 
saveleaveword.php 文件 用 于 对 表单 中 提交 的 数据 进行 处 理 ， 将 数据 存储 到 数据 库 中 。 其 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\22\saveleaveword.php) 


<?php session_start(); /初始 化 一 个 session 变量 
$xym=$_POST[xym]; /获取 $_POST 提交 的 值 
if($xym!=$_SESSION["autonum"]X{ // 淹 断 验证 码 是 否 正 确 
echo "<script>alert( 效 验 码 输入 错误 !);history.back();</script> 
exit; 
8 
Stitle=$_POSTT'title"]; /获取 反 馈 信息 的 标题 
$content=$_POST["content"]; // 获 取 反 馈 信息 的 内 容 
$type=$_POST['"type"]; /获取 反馈 信息 的 类 型 
include_once("conn/conn.php"); // 与 数据 库 建 立 连 接 
$sql=mysql_query("select id from tb_user where usernc=".$_SESSION["unc"]."",$conn); // 读 取 数 据 库 中 数据 
$info=mysql_fetch_array($sql); /获取 结果 集中 的 数组 
$userid=$info["id"]; 
// 向 数据 库 中 添加 数据 


if(mysql_query("insert into tb_leaveword(userid ,type ,title,content,createtime) 
values('$userid','$type','S$title','$content',".date("Y-m-j H:i:s").")",$conn)\{ 


echo "<script>alert(' 留 言 发 表 成 功 !");history.back();</script>"; // 添 加 操作 成 功 ， 给 出 提示 信息 
}else{ 
echo "<script>alert(' 留 言 发 表 失 败 !);history.back();</script>"; 1/ 添 加 操作 失败 ， 给 出 提示 信息 


?> 


22.6 在 线 订 购 模 块 设计 


铭 4 视频 讲解 : 光盘 \TM\Video\ 第 22 章 \ 在 线 订购 模块 设计 .exe 
在 线 订购 模块 的 功能 是 实现 在 线 购买 企业 推出 的 软件 产品 ， 其 操作 主要 通过 购物 车 和 订单 管理 来 实现 。 


22.6.1 在线 订 购 模 块 概述 


在 线 订购 模块 的 功能 对 所 有 访问 网 站 的 人 开放 ， 没 有 任何 权限 限制 。 其 操作 流程 如 图 22.24 所 示 。 


22.24 在线 订 购 模块 的 操作 流程 


22.6.2 


复杂 ， 并 且 要 对 源 文档 中 要 打印 的 内 容 进 行 约束 。 
下 面 介绍 WebBrowser 控件 的 具体 参数 ， 如 表 22.2 所 示 。 
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订单 的 预览 及 打印 技术 


在 线 订 购 管理 模块 中 ， 不 可 缺少 的 一 项 内 容 就 是 对 订单 进行 打印 。 下 面 就 来 讲解 一 下 订单 打印 功能 的 
实现 方法 。 在 线 订 购 管理 模块 中 运用 的 是 WebBrowser 打印 方法 。WebBrowser 是 正 内 置 的 浏览 器 控件 ,无 
须 用 户 下 载 。 其 优点 是 客户 端 独立 完成 打印 目标 文档 的 生成 ， 减 轻 服务 器 负荷 ;缺点 是 源 文 档 的 分 析 操 作 


表 22.2 WebBrowser 控件 的 具体 参数 及 说 明 


4S2 


参数 名 称 说 明 
document .all WebBrowser.Execwb(7.1): 表示 打印 预览 
document.all. WebBrowser.Execwb(6.1): 表示 打印 
document.all. WebBrowser.Execwb(6.6): 表示 直接 打印 
document.all. WebBrowser.Execwb(8.1): 表示 页 面 设置 
document.all.WebBrowser.Execwb(1.1): 打开 页 面 
document.all. WebBrowser.Execwb(2.1): 关闭 所 有 打开 的 正 窗口 
document.all. WebBrowser Execwb(4.1): 保存 网 页 
document.all. WebBrowser.Execwb(10,1): 查看 页 面 属性 
document.all. WebBrowser.Execwb(17,1); 全 选 
document.all. WebBrowser.Execwb(22.1); 刷新 
document.all. WebBrowser.Execwb(45.,1); 关闭 窗 体 无 提示 


该 技术 的 实现 原理 是 : 首先 通过 onClick 事件 调用 一 个 JavaScript 脚本 ， 然 后 执行 openprintwindow0) 函 
数 ， 将 指定 的 变量 值 传递 到 订单 打印 页 (printwindow.php) 中， 最 后 在 订单 打印 页 中 实现 打印 及 打印 预览 节 
功能 。 调 用 JavaScript 脚本 和 执行 openprintwindow0 函 数 在 shopping_dd.php 页 中 完成 。 其 关键 代码 如 下 : 


(代码 位 置 ， 光 盘 \TM\Instance\22\shopping_dd.php》 


<script language="javascript"> 
function openprintwindow(x,y,zX{ 


/定义 一 个 函数 ， 获 取 传 递 的 参数 


/通过 window 对 象 中 的 open() 方 法 打开 一 个 新 窗口 ， 并 设置 其 属性 
@ window.open("printwindow.php?ddno="+x+"&pv="+z,"Nnewframe","top=200,left=200,width=635, 
height="+(230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no,status=no"); 


} 
</script> 
< 上 |- 


通过 onclick 事件 调用 JavaScript 脚本 ， 传 递 参数 值 。 当 参数 z 的 值 为 p 时 执行 打印 功能 


-> 


<td width="75"> <img src="images/bg_14(11)jpg" width="69" height="20" 
@ onclick="javascript:openprintwindow('<?php echo base64_decode($_GET["ddno"]);?>''<?php echo 


S$gnum;?>",'p')" style="cursor:hand"/></td> 


< 上 一 一 通过 onclick 事件 调用 JavaScript 脚本 ， 传 递 参 数值 。 当 参数 z 的 值 为 v 时 执行 打印 预览 功能 


一 > 


<td width="90"><img src="images/bg_14(12)jpg" width="69" height="20" 


onclick="javascript:openprintwindow('<?php 


S$gnum;?>",Vv'")" 
style="cursor:hand"/></td> 


base64_decode($_GET["ddno"]);?>', <?php echo 
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/. 
和 培 胃 @ open0: 打开 一 个 窗口 ， 可 以 设置 其 URL、 名 称 、 大 小 、 按 钮 以 及 其 他 属性 。 该 方法 的 基本 

语法 结构 为 : 

window.open(url,name ,features ,replace) 

参数 url 是 要 在 新 窗口 中 打开 的 文档 的 URL 地 址 ; 参数 name 是 要 打开 的 窗口 的 名 字 ， 用 HIML 链 
接 的 target 属性 进行 定位 时 会 用 到 ; 参数 features 是 一 个 用 过 号 分 隔 的 字符 串 ， 列 举 窗口 的 特征 ; 参数 
Teplace 是 一 个 可 选 的 Boolean 值 ， 指 出 是 否 允 许 URL 替换 窗口 的 内 容 。 

@ onclick: 表明 某 元 素 被 鼠标 单 击 触 发 的 事件 。 


订单 的 打印 和 打印 预览 功能 在 printwindow.php 页 中 完成 , 首先 编写 一 个 实现 打印 预览 功能 的 JavaScript 
脚本 ， 然 后 建立 HTML 的 object 标签 ， 调 用 WebBrowser 控件 ， 最 后 获取 变量 传递 的 值 ， 当 变量 的 值 为 p 
时 执行 打印 功能 ， 当 变量 的 值 为 v 时 执行 打印 预览 的 功能 。printwindow.php 页 的 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\22\printwindow.php) 
<script> 


function printview(){ /定义 一 个 函数 
@ document.all.WebBrowser1.ExecWB(7,1) ; ll 调用 WebBrowser 控件 ， 实 现 打印 预览 
window.close(); /关闭 窗口 
</script> 
< 上 一- 一 一 -一 一 一 一 一 -一 -建立 HTML 的 object 标签 ， 调 用 WebBrowser 控件 -一 一 -一 一 一 -一 一 --> 


<object ID="WebBrowser1' WIDTH=0 HEIGHT=0 
CLASSID='CLSID:8856F961-340A-11D0-A96B-00C04FD705A2'></object> 


< 一 一 一 在 body 中 调用 onload 事件 ， 执 行 打 印 操作 一 一 一 一 > 
四 <bodytopmargin="0" leftmargin="0" bottommargin="0" onLoad=" 
<?php 
if($_GET["pv"]=="p"X{ /| 判断 当 变 量 值 为 p 了 时， 执行 打印 操作 
?> 
© window.print(); 
<?php 
jelseif($_GET["pv]=="v")f // 判 断 当 变 量 值 为 v 时 ， 执 行 打印 预览 操作 
?> 
printview() 


<?php} ?> 
he 


AL/ 
和 培 明 @ document.all. WebBrowserl.ExecWB(7,1): WebBrowser 控件 中 的 参数 ， 表 示 打 印 预 览 。 
@ onLoad: 表明 某 对 象 已 载 入 窗口 。 
@ print0: window 对 象 中 的 一 个 方法 ， 实 现 打印 的 功能 。 


订单 打印 操作 的 运行 效果 如 图 22.25 所 示 。 
22.6.3 ”购物 车 的 实现 过 程 


购物 车 的 功能 是 临时 存储 用 户 选 购 的 商品 ， 用 户 可 以 对 购物 车 中 的 商品 进行 添加 、 修 改 、 删 除 和 更 新 
操作 ， 也 可 以 选择 进行 结算 。 其 运行 的 效果 如 图 22.26 所 示 。 
>) 
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图 22.25 订单 打印 操作 的 运行 效果 图 22.26 购物 车 的 运行 效果 


购物 车 功能 实现 的 第 一 步 是 为 想 要 购买 产品 的 用 户 分 配 一 辆 购物 车 ， 使 其 能 够 记录 自己 已 经 选 购 的 产 
品 。 其 工作 的 原理 与 超市 中 顾客 使 用 购物 车 进行 购物 是 相同 的 ， 只 是 这 里 使 用 的 不 是 真正 意义 上 的 购物 车 ， 
而 是 两 个 session 变量 ， 一 个 存储 用 户 选 购 商品 的 ID (S$goodsid)〉 ， 另 一 个 存储 用 户 选 购 该 商品 的 数量 
($goodsnum) 。 如 果 用 户 在 一 次 购物 中 选 购 多 种 不 同类 的 商品 ， 则 使 用 “@” 对 不 同类 商品 的 不 同 ID 和 
数量 进行 分 隔 。 例 如 , 用 户 选 购 的 不 同类 商品 id 为 1、2、3, 则 session 变量 $goodsid 中 存储 的 值 为 “1@2@3@”。 
其 中 同一 种 商品 不 能 购买 两 次 ， 如 果 想 要 购买 多 个 同 种 产品 ， 可 以 在 购物 车 中 更 改 购买 商品 的 数量 。 在 本 
项 目 中 , 购物 车 的 分 配 功能 通过 shopping_cart_first.php 文件 来 完成 。 shopping_cart_first.php 文件 的 代码 如 下 : 

〈 代 码 位 置 ， 光盘 \TMNInstance\22\shopping_cart_firstphp) 


@ <?php 。 session_start(); /初始 化 session 变量 

@ session_register("goodsid"); /创建 一 个 session 变量 

© session_register("goodsnum"); /创建 一 个 session 变量 

if($_SESSION["goodsid"]==" && $_SESSION["goodsnum"]==""){ ”// 判 断 session 变量 中 的 值 是 否 为 空 
$_SESSION['goodsid]=$_GET[T'id]"@": // 如 果 为 空 则 将 商品 的 ID 赋 给 变量 
$_SESSION["goodsnum"]="1@"; // 将 商品 数量 设置 为 1@ 

Jelse{ 

9 $array=explode("@",$_SESSION["goodsid"]); // 如 果 不 为 空 ， 则 使 用 @ 分 隔 不 同 的 商品 ID 


// 判 断 如果 获 取 的 ID 在 session 变量 中 已 经 存在 ， 则 提示 该 商品 已 经 被 放 入 购物 车 
© ifin_array($_GET["id"],$array)){ 
echo "<script>alert(' 该 编程 词典 已 经 被 放 入 购物 车 ! ');history.back();</script>"; 


exit; 
上 
$_SESSION["goodsid"].=$_GET["id"]."@"; /为 session 变量 赋值 
$_SESSION["goodsnum"].="1@"; /为 session 变量 赋值 


} 

// 将 商品 放 入 购物 车 中 ， 并 跳 转 到 购物 车 页 

echo "<script>window.location.href='shopping_cart.php';</script>"; 
?> 


说 日 
Nm 明 @ session start0: 初始 化 session 变量 。 
@ session register(): 创建 一 个 session 变量 goodsid， 存 储 商 品 人 D。 
上 @ session register0: 创建 一 个 session 变量 goodsnum， 存 储 购买 商品 数量 。 
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@ explode0: 将 字符 囊 依 指 定 的 字符 囊 或 字符 进行 分 隔 。 返 回 由 字符 囊 组 成 的 数组 ， 每 个 元 素 都 是 
string 的 一 个 子囊 ， 它 们 被 字符 囊 separator 作为 边界 点 分 隔 出 来 。 如 果 设置 了 limit 参数 ， 则 返回 的 数组 
包含 最 多 limit 个 元 素 , 而 最 后 那个 元 素 将 包含 string 的 剩余 部 分 ; 如 果 separator 为 空 字符 囊 (""),explode() 
通 数 将 返回 false; 如 果 separator 所 包含 的 值 在 string 中 找 不 到 ， 那 么 explode0 函 数 将 返回 包含 string 个 
元 素 的 数组 ; 如 果 参 数 limit 是 负数 ， 则 返回 除了 最 后 的 -limit 个 元 素 外 的 所 有 元 素 。 

@ in array0: 在 指定 的 数组 中 搜索 某 个 值 ， 如 果 找 到 则 返回 tue， 否 则 返回 false。 


在 实现 购物 车 的 分 配 和 添加 商品 的 功能 后 ， 接 下 来 要 做 的 就 是 查看 购物 车 中 的 商品 ， 即 实现 购物 车 中 
商品 展示 的 功能 。 在 购物 车 的 商品 展示 中 ， 可 以 实现 清空 购物 车 、 删 除 购买 商品 、 更 改 购买 商品 数量 、 继 
续 购 物 和 结算 等 操作 。 

购物 车 商品 展示 的 功能 主要 通过 shopping_cart.php 文件 来 完成 , 首先 从 session 变量 中 读 取 商品 的 ID 和 
数量 ， 然 后 根据 商品 的 ID 循环 输出 购物 车 中 的 商品 ， 最 后 以 商品 ID 为 标识 符 设置 不 同 的 超 链 接 ， 执 行 删 
除 商 品 或 者 更 改 购买 商品 数量 等 操作 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\InstanceW22\shopping_cart.php) 

<table width="720" height="47" border="0" align="center" cellpadding="0" cellspacing="1" bgcolor="#999999"> 

<?php 


$array=explode("@",$_SESSION["goodsid"]); // 读 取 session 变量 中 的 商品 ID， 以 @ 进 行 分 隔 
$arraynum=explode("@",$_SESSION["goodsnum"]); // 读 取 session 变量 中 的 商品 数量 ， 以 @ 进 行 分 隔 
$markid=0; // 创 建 变量 ， 初 始 值 为 0 
for($i=0;$i<count($array):$i++}{ /应 用 for 循环 语句 循环 输出 商品 ID 的 值 

if($array[$i]!="™"}{ /| 判断 如 果 商 品 ID 的 值 不 为 空 

$markid++; /增加 变量 $markid 的 值 

} 
! 
if($markid==0X{ // 判 断 如 果 变 量 $markid 的 值 为 空 则 输出 下 面 的 内 容 


?> 


<tr> 
<td height="22" colspan="4”bgcolor="#FFFFFF"><div align="center"> 对 不 起 您 的 购物 车 中 暂 无 商品 信 
息 !</div></td> 


</tr> 
<?php 
}else{ // 如 果 $markid 的 值 不 为 空 ， 则 执行 下 面 的 内 容 
S$totalprice=0; // 创 建 变量 $totalprice， 初 始 值 为 0 
for($i=0;$i<count($array):$i++){ // 循 环 输出 数组 中 的 商品 ID 值 
if($array[$i]!="}{ 
/根据 获取 的 商品 ID 的 值 ， 从 数据 库 中 获取 对 应 产品 的 信息 
$sqlcart=mysql_query("select * from tb_bccd where id=".$array[$i]."",$conn); 
Sinfocart=mysql_fetch_array($sqlcart) 
2% 
<tr> 


<form name="form<?php echo $array[$i]?>" method="post" action="changegoodsnum.php"> 

<td height="22" bgcolor="#FFFFFF">&nbsp;<?php echo unhtml($infocart["bccdname"]);?></td> 

<td height="22" bgcolor="#FFFFFF"><div align="center"><?php echo 
number_format($infocart["price"],2);?></div></td> 

<td height="22" bgcolor="#FFFFFF"><div align="center"><input type="text" name="goodsnum" 
value="<?php echo $arraynum["$i"];?>" class="inputcss" size="8" ><input type="hidden" name="id" 
value="<?php echo 
Sinfocart["id"]:?>" ></div></td> 

<td height="22" bgcolor="#FFFFFF"><div align="center"><a href="javascript:form<?php echo 
$arrayf$j?>.submit()" class="a1"> 更 改 数量 </a>&nbsp;|&nbsp;<a href="delgoods.php?id=<?php echo Sinfocart["id"]:?>" 
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class="a1"> 删 除 该 项 </a></div></td> 


</form> 
</tr> 
<?php 
Stotalprice+=S$infocart["price"]*$arraynum["$i"]; 
} 
} 
y 
ys 
</table> 


22.6.4 商品 订单 的 实现 过 程 


在 确定 所 要 购买 的 商品 后 ， 接 下 来 要 做 的 就 是 进行 购物 结算 ， 填 写 用 户 购物 订单 ， 将 订单 保存 到 数据 
库 中 ， 并 且 随 机 生成 一 个 订单 号 ， 作 为 订单 的 唯一 标识 。 生 成 订单 的 运行 效果 如 图 22.27 所 示 。 
(OEE 


让 间作 明 上 和 二 人 于 请 各 河和 可 单 


印章 ， [本 号 元 
介 本 要 支付 的 全 大 避 计 为 :1a0 下 元 


i 后 汪 本 


图 22.27 生成 订单 的 运行 效果 

提交 订单 后 ， 用 户 可 以 选择 汇款 的 方式 : 一 是 选择 网 上 支付 ， 那 么 将 跳 转 到 企业 指定 的 网 上 银行 进行 
汇款 ， 汇 款 的 操作 将 在 企业 指定 的 网 上 银行 中 进行 ， 这 里 不 做 讲解 ， 二 是 选择 到 指定 的 银行 向 企业 提供 的 
账号 中 汇款 。 企 业 将 在 收 到 汇款 后 按照 用 户 指定 的 地 址 和 方式 配送 产品 。 商 品 订 单 的 生成 和 处 理由 
shopping_cart_getuserinfo .php 和 savebuyuser.php 文件 来 完成 。 

订单 处 理由 savebuyuser.php 文件 完成 ， 首 先 连接 数据 库 ， 随 机 生成 一 个 订单 号 ， 然 后 获取 购物 车 中 的 
商品 信息 ， 最 后 将 商品 信息 和 订单 号 存储 到 数据 库 中 。 其 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\22\savebuyuser.php) 


<?php session_start(); /| 初始 化 session 变量 
include_once("conn/conn.php"); /| 连接 数据 库 
$ddnumber=substr(date("YmdHis"),2,8).mt_rand(100000,999999); // 随 机 生成 订单 号 


$sql=mysql_query("select * from tb_city where id=".$_POST["city]."",$conn); // 读 取 数 据 库 中 的 城市 信息 
$info=mysql_fetch_array($sql); 
if($shfs=="1"X{ 1/ 判断 用 户 选择 的 送 货 方 式 
$yprice=$info[ pt]; 
$shfs=" 普 通 邮递 "; 
jelseif($shfs=="2"){ 
$yprice=$info'[kd]; 
Sshfs=" 邮 政 特 快 专递 EMS"; 
» 
$array=explode("@",$_SESSION["goodsid"]); /省 以 @ 来 分 隔 session 变量 中 存储 的 商品 ID 
$arraynum=explode("@",$_SESSION["goodsnum"]); /以 @ 来 分 隔 session 变量 中 存储 的 商品 数量 
S$totalprice=0; 
for($i=0;$i<count($array):$i++){ /| 循环 读 取 数组 中 商品 的 ID 
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if($array[$i]!="™"){ 
S$sqlcart=mysql_query("select * from tb_bccd where id=". $array[$i]."",$conn); 
$infocart=mysql_fetch_array($sqlcart): 
Stotalprice+=S$infocart["price"]*$arraynum["$i"]; 

} 


} 

Stotalprice=S$totalprice+ $yprice; 1/ 获取 汇款 金额 

// 将 表单 中 提交 的 数据 存储 到 数据 库 中 

if(mysql_query("insert into 
tb_dd(ddnumberrecuser,sex,address,yb,qq,email,mtel,gtel,shfs,spc,slc,yprice,totalprice,createtime,cityid) 


_POST["qq"].",".$_POSTr'email"].",".$_POSTT'mtel"].",".$_POSTr'gtel"].” 
™",".$_SESSION["goodsnum"].",”".$yprice.",". $totalprice.”",".date("Y-m-d Hi 


"$shfs.",".$_SESSION["goodsid"] 
",".$_POSTT'city"].")", $conn) 


session_unregister("goodsid"); // 注 销 session 变量 goodsid 
session_unregister("goodsnum"); /注销 session 变量 goodsnum 


echo "<script>window.location.href='shopping_dd.php?ddno=".base64_encode($ddnumber).";</script>"; 
}else{ 

echo "<script>alert(' 订 单 信息 保存 失败 ， 请 重 试 !);</script>"; 
上 


?>: 
22.7 社区 论坛 模块 设计 
锅 4 视频 讲解 : 光盘 \TM\Video\ 第 22 章 \ 社 区 论坛 模块 设计 .exe 
社区 论坛 模块 为 网 站 的 浏览 者 提供 一 个 交流 的 平台 ， 以 此 来 扩大 网 站 的 影响 力 ， 汇 聚 更 多 的 人 气 ， 宣 
传 企业 形象 ， 推 广 企业 产品 。 


22.7.1 社区 论坛 模块 概述 


产 


区 论坛 模块 为 浏览 者 、 会 员 、 客 户 和 企业 提供 了 一 个 
大 的 交流 平台 ， 根 据 身份 的 不 同 ， 给 予 不 同 的 操作 权限 ， 社 
区 论坛 模块 的 操作 流程 如 图 22.28 所 示 。 

在 本 论坛 中 ， 浏 览 者 只 能 够 查看 帖子 ， 注 册 会 员 既 可 以 
查看 帖子 ， 也 可 以 发 布 和 回复 帖子 ;管理 员 则 具有 发 布 、 回 
复 、 查 看 和 删除 帖子 的 权限 。 


22.7.2 页面 跳 转 技 术 


在 社区 论坛 模块 的 实现 过 程 中 ， 通 过 JavaScript 脚本 和 
下 拉 列 表 框 的 结合 实现 一 个 不 同 版 块 之 间 快 速 跳 转 的 功能 ， 
从 而 能 够 更 加 灵活 、 方 便 地 实现 不 同 版 块 之 间 的 跳 转 。 

下 面 分 析 该 技术 是 如 何 实现 的 。 该 技术 的 实现 综合 3 个 
方面 的 内 容 ， 以 一 个 下 拉 列 表 框 为 主 ， 通 过 PHP 语句 从 数据 库 中 读 取 数据 作为 下 拉 列 表 框 的 值 ， 应 用 
onchange 事件 来 调用 JavaScript 脚本 ， 实 现 不 同 版 块 之 间 的 跳 转 。 这 里 以 bbs_top.php 文件 中 的 快速 跳 转 功 


能 为 例 进行 分 析 。 其 关键 代码 如 下 : 
@ 


图 22.28 社区 论坛 操作 流程 图 
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〈 代 码 位 置 : 光盘 \TMNVInstance\22\bbs_top-php) 
<!- 创 建 一 个 下 拉 列 表 框 ， 指 定名 称 为 select_type， 并 且 设 置 其 属性 ， 通 过 onchange 事件 来 调用 JavaScript 脚 
本 文件 ， 实 现 页 面 跳 转 一 > 
<select name="select_type" class="inputcss" 
@ onChange="javascript:window.location=this.options[this.selectedindex].value;" > 


<?php 
i aii hot ili 使 用 数据 的 ID 作为 下 拉 列 表 框 的 值 , 使 用 数据 的 标题 title 作为 下 拉 列 表 框 
示 | 
$sql=mysql_query("select * from tb_type_small order by createtime desc",$conn); 
$info=mysql_fetch_array($sql); 


if($info=="™"){ 
echo "<option> 暂 无 讨论 区 </option>"; 
Jelse{ 
echo "<option>- 版 块 快速 跳 转 -</option>"; 
dof /应 用 do…while 循环 语句 输出 下 拉 列 表 框 中 的 值 
@ echo "<option value='bbs_list.php?id=".$info['id"].">".$infof'title]."</option>"; 
) 
while($info=mysql_fetch_array($sql)); /应 用 do…while 循环 语句 结束 
?> 
</select> 
Cm 
@ onChange: 某 元 素 失去 焦点 ， 并 且 从 用 户 最 后 一 次 访问 以 来 ， 其 值 已 经 改变 。 


location: 用 于 访问 窗口 的 当前 定位 (URL) ， 既 可 被 读 取 ， 又 可 被 置换 ， 可 以 通过 其 实现 某 个 页 面 
的 定位 或 者 更 新 。 
@ <option value='… >.…</option>: 下 拉 列 表 框 中 输出 的 值 ， 以 及 显示 的 内 容 。 


该 技术 实现 的 效果 如 图 22.29 所 示 ， 它 将 实现 从 硬件 咨询 模块 跳 转 到 PHP 模块 。 


22.7.3 ”论坛 分 类 的 实现 过 程 


论坛 分 类 实现 两 个 功能 ， 一 是 实现 论坛 中 大 的 版 块 分 区 ， 分 为 6 个 版 块 : 综合 信息 讨论 区 、 操 作 系统 、 
程序 设计 交流 区 、 网 管 技术 应 用 、Web 程序 开发 和 数据 库 技 术 ， 其 数据 存储 于 tb_type_big 数据 表 中 。 另 一 
个 是 不 同 版 块 中 不 同 语言 和 技术 的 分 类 ， 分 为 11 种 ， 其 数据 存储 于 tb_type_small 表 中 。 论 坛 分 类 的 运行 效 
果 如 图 22.30 所 示 。 


C3 必 村 而。 由 本 


EECTS 


EREET 


图 22.30 论坛 分 类 的 运行 效果 
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论坛 分 类 的 实现 原理 很 简单 ， 首 先 从 tb_type_big 表 中 读 取 6 个 版 块 中 的 数据 ， 进 行 循 环 输出 ， 然 后 在 
版 块 中 嵌 套 循环 , 用 于 输出 不 用 语言 的 分 类 数据 。 该 功能 主要 通过 bbs_index.php 文件 来 完成 。 bbs_index.php 
文件 的 程序 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\InstanceW22\bbs_index.php) 


<?php include_once("top.php"); // 调 用 网 站 头 文件 
include_once("bbs_top.php"); /1 调用 社区 论坛 的 头 文件 

?> 

<?php 


1/ 循环 输出 数据 表 tb_type_big 中 的 6 个 版 块 数据 
$sql=mysql_query("select * from tb_type_big order by createtime desc",$conn); 
$info=mysql_fetch_array($sql); 


if($info==false}{ // 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 
了 
// 省 略 了 部 分 代码 
<?php 
jelsef // 如 果 返 回 值 为 true， 则 执行 do…while 循环 语句 
上 外 部 嵌 套 循环 ， 输 出 论坛 中 的 版 块 分 类 数据 */ 
0 dof 
?> 
/省 略 了 部 分 代码 


<table width="750" border="0" align="center" cellpadding="0" cellspacing="1" bordercolor="#FFFFFF" 
bgcolor="#6EBEC7"> 
<?php 
/循环 输出 tb_type_small 表 中 的 不 同 语言 和 技术 的 分 类 数据 
$sql1=mysql_query("select * from tb_type_small where bigtypeid=".$info["id"]."",$conn); 
$info1=mysql_fetch_array($sql1); 


if($info1==falseX{ // 判 断 如 果 返 回 值 为 false， 则 执行 下 面 的 内 容 
?> 
于 // 省 略 了 部 分 代码 
<?php 
}else{ /如 果 返 回 值 为 rue， 则 执行 下 面 的 内 容 ， 输 出 该 版 块 中 对 应 语言 和 技术 帖子 的 详细 信息 
六 
Ee // 省 略 了 部 分 代码 
@ ” <?php ”/* 内 部 柑 套 循环 输出 不 同 语言 和 技术 的 分 类 */ 
do{ ?> 
<tr> 
<td height="30"><font color="#666666"> 创 建 时 间 :<?php echo Sinfo1["createtime"];?></font></td> 
</tr> 
// 省 略 了 部 分 代码 
<?php 
jwhile($info1=mysql_fetch_array($sql1)); 
/* 内 部 嵌 套 循环 结束 */ 
> 
</table> 
<?php 
}while($info=mysql_fetch_array($sq)l)); 
上 外 部 嵌 套 循环 结束 ， 对 版 块 中 的 大 类 进行 输出 */ 
i 
<?php include_once("bottom.php"); 2 


oh. 
下 培 明 @ do…while 循环 语句 ， 对 论坛 中 大 的 版 块 分 类 进行 循环 输出 。 
@ do…while 循环 语句 ， 对 论坛 中 一 个 版 块 中 的 不 同 语 言 和 技术 进行 循环 输出 。 
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人 注意 在 应 用 do…while 循环 语句 时 ，while 后 的 分 号 不 能 省 略 。 


22.7.4 ”论坛 帖子 浏览 的 实现 过 程 


论坛 帖子 浏览 主要 输出 指定 帖子 的 详细 信息 , 包括 发 
帖 人 、 用 户 级 别 、 注 册 的 时 间 ， 以 及 帖子 的 主题 、 内 容 和 
发 帖 时 间 , 包括 上 传 的 图 片 。 本 模块 中 是 用 户 权限 使 用 体 
现 的 最 明显 的 地 方 , 可 以 分 为 3 种 情况 : 以 浏览 者 进行 登 
录 ， 只 能 浏览 帖子 的 内 容 ， 没 有 其 他 权限 : 以 会 员 身份 进 
行 登录 ， 可 以 对 帖子 进行 回复 ， 发 表 自己 的 看 法 ;以 管理 
员 的 身份 进行 登录 , 不 但 可 以 回复 帖子 , 而 且 可 以 对 任何 
人 发 布 和 回复 的 帖子 进行 删除 和 顶 帖 的 操作 。 以 管理 员 身 
份 进行 登录 时 的 运行 效果 如 图 22.31 所 示 。 Ee 

论坛 帖子 浏览 的 功能 通过 bbs_lookbbs.php 文件 完成 ， 图 22.31 管理 员 浏 览 帖子 的 结果 图 
首先 根据 传递 的 ID 值 读 取 指定 的 帖子 数据 ， 然 后 判断 登 
录用 户 的 类 型 ， 最 后 根据 用 户 不 同 的 类 型 执行 不 同 的 操作 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TMNInstance\22\bbs_lookbbs-php) 

<?phl 

/根据 $_GET 传递 的 数据 获取 tb_bbs 中 的 数据 

$sqlb=mysql_query("select * from tb_bbs where id=".$_GET["id"]."",$conn); 

$infob=mysql_fetch_array($sqlb); 

/根据 $_GET 传递 的 数据 获取 tb_user 中 的 数据 

$sql4=mysql_query("select * from tb_user where id=".$infob["userid"]."",$conn); 

$info4=mysql_fetch_array($sql4); 


?> 
/省 略 了 部 分 HTML 代码 
ble width="180" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 


Om sa i 


HH 


<td height="22"> 用 户 级 别 : 
<?php 
/根据 用 户 信息 表 tb_user 中 字段 usertype 的 值 判断 该 用 户 的 类 型 
// 如 果 值 为 1 则 是 管理 员 ， 值 为 2 则 是 后 台 管 理 员 ， 值 为 0 则 是 普通 会 员 
@ if($info4["usertype"]=="1") echo "管理 员 ";else echo "普通 会 员 "; 
> 
</td> 
</tr> 
</table> 
加 /省略 了 部 分 HTML 代码 
<table width="500" height="200" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 


<td width="35" height="20"><div align="center"><img src="images/lt_15(11).jpg" width="25" 
height="25"></div></td> 
<td><?php echo S$infob["createtime"];?></td> 
</tr> 
<tr> 
<td height="150" colspan="2"> 
<?php 
// 判 断 tb_bbs 表 中 的 字段 photo 是 否 为 空 ， 为 空 则 执行 下 面 的 内 容 
if($infob[photo]i="”){f 
S$photos=substr($infob['photo'],2,70); 1/ 获取 图 片 在 服务 器 中 的 存储 路 径 


局 
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echo (stripslashes($infob["content"])); // 输 出 帖子 的 内 容 


echo "<img src=\"$photos\">"; /根据 获取 的 图 片 路 径 ， 输 出 服务 器 中 的 图 片 
jelsef /如 果 tb_bbs 表 中 的 图 片 字段 photo 为 空 ， 则 执行 下 面 的 内 容 
目 echo (stripslashes($infob["content])); /只 输出 帖子 的 内 容 
?> 
</td> 
</tr> 
</table> 


<table width="530" height="20" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 


<td width="239" height="30">&nbsp;</td> 
<td width="291"><img src="images/lt_15(5).jpg" width="72" height="23" style="cursor:hand" onclick=" 
<?php 

/如 果 $_SESSION["unc"] 的 值 为 空 ， 则 不 可 以 进行 项 帖子 的 操作 

if($_SESSION["unc"]==" 
echo "javascript:alert(' 请 先 登录 本 站 ， 然 后 进行 此 操作 ! ');window.location.href='index.php""; 

jelse{ 
// 否 则 将 判断 当前 用 户 的 类 型 ， 如 果 是 管理 员 则 可 以 顶 帖 
$sqlu=mysql_query("select usertype from tb_user where usernc=".$_SESSION["unc"]. 


"",$conn); 
$infou=mysql_fetch_array($sqlu); 
[3 if($infou["usertype"]==1X{ /| 如果 用 户 的 类 型 为 1， 则 有 项 帖 的 权限 
echo "javascript:window.location.href='settop.php?id=".$infob["id"].”"; 
}else{ // 否 则 不 具备 该 权限 
echo "javascript:alert(' 对 不 起 ， 您 不 具备 该 操作 权限 ! ");"; 
; 
?> 
"/>&nbsp;&nbsp; 
© <?php 


// 判 断 当前 用 户 是 否 具有 删除 帖子 的 权限 

if($_SESSION["unc"]!=™"}{ 
// 条 件 为 用 户 不 能 为 空 ， 并 且 是 管理 员 ， 才 具备 删除 帖子 的 权限 
$sqlu=mysql_query("select usertype from tb_user where 

Usernc=".$_SESSION["unc"]."",$conn); 

Sinfou=mysql_fetch_array($sqlu); 

if($infou["usertype"]==1X{ 
?> 


<img src="images/lt_15(10).jpg" onclick="javascriptif(window.confirm(' 您 确定 删除 该 帖 吗 ? 
")==true){window.location.href='bbs_delete.php?id=<?php echo $infob["id"]?>";}" style="cursor:hand"/> 
<?php 


?> 
</td> 
</tr> 
</table> 


a 
EC 涪 明 @ $info4["usertype"]: 判断 用 户 的 类 型 ， 如 果 值 为 1 是 管理 员 ， 否 则 为 普通 会 员 。 
@ $infob['photo']: 判断 发 布 的 帖子 中 是 否 含有 图 片 ， 如 果 有 则 输出 ， 没 有 则 不 输出 。 
@ stripslashes(): 将 应 用 addcslashes0 函 数 处 理 后 的 字符 串 按 原样 返回 。 
@ 判断 登录 用 户 是 否 具有 顶 帖 的 权限 。 
@ 判断 登录 用 户 是 否 具有 删除 帖子 的 权限 。 
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22.7:5 


论坛 帖子 发 布 的 实现 过 程 


论坛 帖子 发 布 通过 两 个 文件 来 完成 ， 一 个 是 帖子 发 布 信息 的 提交 页 bbs_pubs.php， 另 一 个 是 对 提交 的 数 
据 进行 处 理 的 retrieve.php 文件 。 帖 子 发 布 模块 的 运行 效果 如 图 22.32 所 示 。 


SNP :1s 
薛 入 国 nick 旺 rt 让 


图 22.32 ”帖子 发 布 模块 的 运行 效果 


在 发 布 信息 的 提交 页 中 ， 显 示 当 前 用 户 的 个 人 信息 ， 设 置 添 加 数据 表单 元 素 ， 其 中 表单 元 素 的 设计 如 


表 22.3 所 示 。 


表 22.3 发布 信 息 页 中 使 用 的 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 
form bbs | _ form method="post" action="retrieve.php" enctype="multipart/form-data" 发 帖 表单 
class="inputcss" style="background-color:#6EBEC7"> 
<?php 
$sql=mysql_query("select * from tb_type_small order by createtime desc",$conn); 
Sinfo=mysql fetch_array($sqD: 
这 $info 一 false){ 本 
echo "<option> 暂 无 讨论 区 </option>"; 选择 发 表 
bbs select eol 向 的 
a > 言 或 者 技 
<option value="<?php echo $info['id] :?>"<?php 这 $_GET[id] 一 $infof'id]) 术 的 类 别 
{echo "selected=\"selected\"":}?>><?php echo $info['title']:?></option> 
<?php } 
while($info=mysql fetch_array($sq))): 
js 
bbs title text class="inputcss" style="background-color#6EBEC7"> 帖子 标题 
bbs head Tadio value="<?php echo("images/bbsface/face".($i-1).".gif"):?>" 表情 图 
bbs_photo file id="bbs_photo" class="inputcss" style="backeround-color:#6EBEC7" /> 上 传 图 片 
contentl textarea _ |id="contentl" class="inputcss" style= background-color#6EBEC7"> 帖子 内 容 
Submit submit Value=" 提 交 " 提交 表单 
在 retrieve php 中 对 表单 提交 的 数据 进行 处 理 ， 将 数据 存储 到 tb_bbs 表 中 ,并 且 更 新 用 户 信息 表 tb_user 


中 pubtimes 字段 的 值 ， 其 中 还 应 用 了 图 片上 传 技术 ， 将 图 片上 传 到 服务 器 中 指定 的 文件 夹 下 。retrieve.php 
文件 的 代码 如 下 : 
《代码 位 置 ， 光盘 \TM\Instance\22\retrieve.php) 


局 
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<?php session_start(); /初始 化 session 变量 
Stitle=$_POST[bbs title]: // 获 取 帖子 的 标题 
$content=$_POST[content11; /获取 帖子 的 内 容 
/* 判断 提交 的 帖子 主题 和 帖子 内 容 是 否 为 空 "/ 
if($title=="™"X{ 
echo "<script>alert(' 请 输入 帖子 主题 ');history.back();</script>"; 
exit; } 
if($content=="™"){ 
echo "<script>alert(' 请 输入 帖子 内 容 ");history.back();</script>"; 
exit; } 
J 
include_once("conn/conn.php"); /| 连接 数据 库 
// 根 据 $_SESSION["unc"] 的 值 读 取 数 据 库 中 用 户 的 信息 
$sql=mysql_query("select * from tb_user where usernc=".$_SESSION["unc"]."",$conn); 


$info=mysql_fetch_array($sql); /检索 指定 条 件 的 数据 信息 
$userid=$infor'id]; // 获 取 用 户 id 
$typeid=$_POST[bbs_type]; /接收 版 块 名 称 
Stitle=$_POST[bbs title]; /接收 帖子 主题 
$content=$_POST[content11]; /接收 帖子 内 容 
S$head=$_POST[bbs_head'; /接收 头像 
$createtime=date("Y-m-j H:i:s"); // 获 取 系 统 当前 时 间 
$lastreplytime=$createtime; /将 当前 时 间 赋 给 变量 


$readtimes=0; 
ae ); 
访 $_FILES[bbs_photo']["name"]==true){ /上 传 图 片 ， 判 断 文 件 是 否 存在 ， 如 果 存 在 则 执行 下 面 的 内 容 
区 S$photo_name=strtolower(stristr($_FILES["bbs_photo"]["name"],".")); /获取 图 片 后 缀 名 ， 将 字符 转 成 小 写 
说 $photo_namel=".gif' & $photo_name!=".jpg" & $photo_name!=".jpeg" { // 判 断 图 片 的 格式 是 否 符合 要 求 
echo "<script>alert(' 您 上 传 的 图 片 格式 不 正确 !);history.back();</script>"; 
}else{ 
日 Spaths1=$link.mt_rand(1000000,9999999).$photo_name; 1/ 创建 图 片 的 名 称 
S$photos="./upfile/".$paths1; // 创 建 图 片 的 存储 路 径 
(32 move_uploaded_file($_FILES[bbs_photo"]["tmp_name"],$photos);”// 将 图 片 存储 到 指定 的 文件 夹 下 
// 向 数据 库 添加 数据 
if(mysql_query("insert into tb_bbs(userid,typeid ,title,content,createtime,lastreplytime,head ,readtimes, 
top,photo 
re 
.",'0','$photos')", $conn))\{ 
mysql_query("update th_user set pubtimes=pubtimes+1",$conn);，// 更 新 tb_user 中 pubtimes 字段 的 值 
echo "<script>alert( 新 帖 发 表 成 功 !");history.back();</script>"; 


Jelse{ 
echo "<script>alert( 新 帖 发 表 失 败 !");history.back();</script>"; 
} 
} 
Jelse{ // 如 果 没 有 提交 图 片 ， 则 执行 下 面 的 内 容 


if(mysql_query("insert into tb_bbs(userid,typeid ,title,content,createtime,lastreplytime,head,readtimes,top) 
| 
a mysql_query("update tb_user set pubtimes=pubtimes+1",$conn); 
echo "<script>alert(' 新 帖 发 表 成 功 !");history.back();</script>"; 
1 "<script>alert(' 新 帖 发 表 失 败 !");history.back();</script>"; 
} 


人 
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/ 
和 培 明 @ $_FILES['bbs photo"]["name"]: $_FILES[] 全 局 变量 ， 获 取 表 单 提交 文件 的 原始 名 称 。 
@ strtolower(): 将 指定 的 字符 转换 为 小 写字 母 。 
stristr0: 获取 指定 字符 串 (A ) 在 另 一 个 字符 串 (B) 中 首次 出 现 的 位 置 到 (B ) 字符 囊 末 尾 的 所 有 
字符 串 。 该 函数 如 果 执 行 成 功 则 返回 剩余 的 字符 串 ， 否 则 将 返回 false。 
@ mt rand0: 生成 一 个 随机 数 ， 用 于 上 传 文件 的 名 称 。 
@ move uploaded file(): 将 指定 的 文件 上 传 到 指定 的 文件 夹 下 。 


22.7.6 ”论坛 帖子 回复 的 实现 过 程 


回复 论坛 中 的 帖子 ， 必 须 是 以 会 员 或 者 管理 员 的 身份 进行 登录 ， 否 则 不 能 进行 帖子 的 回复 操作 ， 其 运 
行 效果 如 图 22.33 所 示 。 


Ms 
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22.33 ”论坛 帖子 回复 的 运行 效果 


论坛 帖子 回复 功能 主要 通过 bbs_looks.php 和 savereply.php 文件 实现 。 其 中 应 用 JavaScript 脚本 对 回复 
帖子 的 文本 框 进行 输出 和 隐藏 的 控制 。 在 bbs_looks.php 文件 中 ， 帖 子 回复 使 用 的 表单 元 素 如 表 22.4 所 示 。 


表 22.4 论坛 帖子 回复 中 的 重要 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含 义 
form reply | pm | method="post" action="savereply.php" enctype="multipart/form-data"> | 回复 表单 
Teply title | text | class="inputcss" id="reply_title" | 回复 帖子 主题 
bbsid hidden Value="<?php echo $infob["id"]:?>" 对 应 帖子 的 ID 
bbs head Tadio Value="<?php echo("images/bbsface/face".($i-1).".gif"):?>" 表情 图 
bbs_photo | file | id="bbs_photo" class="inputcss" | 上 传 图 片 
contentl textarea | id="contentl” | 回复 帖子 内 容 
Submit submit value=" 提 交 " 提交 表单 


在 帖子 回复 表单 bbs_looks.php 页 中 ， 首 先 判断 登录 用 户 是 否 具有 回复 的 权限 ， 然 后 根据 提交 的 值 展开 
回复 表单 的 文本 框 ， 在 文本 框 中 输入 回复 的 主题 和 内 容 ， 最 后 将 数据 提交 到 表单 处 理 页 savereply.php 中 。 
bbs_ looks php 的 主要 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\InstanceW22\bbs_lookbbs.php) 


局 
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<script language="javascript"> 


/设计 回复 帖子 表格 的 输出 方式 
function show_reply(X{ /定义 一 个 函数 
if(reply_bbs1.style.display=="™") /| 判断 当 display 的 值 为 空 时 
* 
reply_bbs1.style.display="none" ; // 则 输出 表格 
button_show_bbs.value=" 回 复 帖 子 "; // 显 示 “ 回 复 帖 子 ” 
else if(reply_bbs1.style.display=="none") /判断 当 display 的 值 为 none 时 
{ 
reply_bbs1.style.display="" ; // 则 不 输出 表格 
button_show_bbs.value=" 关 闭 窗口 "; // 显 示 “ 关 闭 窗口 ” 
} 
</script> 
< 上 -一 一 一 一 一 一 一 一 一 一 一 判断 登录 用 户 是 否 具 有 回复 的 权限 一 一 一 一 一 一 一 一 一 一 一 一 一 > 
<img src="images/lt_15(9).jpg" width="72" height="23" id="button_show_bbs" style="cursor:hand" onClick=" 
<?phl 


p 
if($_SESSION["unc"]=="X{ 
echo "javascript:alert(' 请 先 登录 本 站 ， 然 后 回复 帖子 ");window.location.href='index.php';"; 


}else{ 
?> 
show_reply() 
<?php } 人 > fe 


表单 处 理 页 savereply.php 将 表单 提交 的 数据 存储 到 指定 的 数据 库 中 ， 其 实现 的 方法 与 论坛 发 布 中 的 表 
单 处 理 技术 是 相同 的 ， 有 关 该 技术 的 详细 讲解 请 参考 22.7.5 节 ， 这 里 不 再 袭 述 。 


22.8 后台 首页 设计 
铭 i 视频 讲解 ， 光盘 \TM\Video\ 第 22 章 \ 后 台 首 页 设计 .exe 
作为 一 个 完整 的 网 上 社区 系统 ， 要 想 能 够 及 时 地 对 网 站 进行 管理 和 维护 ， 必 须 具 有 一 个 强大 的 后 台 管 
理 系 统 ， 对 网 上 社区 系统 中 的 数据 进行 更 新 和 维护 。 


22.8.1 后 台 首 页 概述 


网 上 社区 系统 的 后 台 管理 采用 的 是 一 种 简单 的 框架 结构 ， 通 过 switch 语句 来 实现 。 其 具体 内 容 如 下 。 
软件 试用 管理 : 包括 软件 试用 产品 的 添加 和 删除 。 

编程 词典 管理 : 包括 编程 词典 版 本 和 内 容 的 添加 、 删 除 。 

在 线 订 购 管理 : 主要 用 于 管理 用 户 提交 的 订单 。 

软件 升级 管理 : 包括 升级 包 和 序列 号 的 添加 、 删 除 。 

站 内 公告 管理 : 主要 用 于 添加 和 删除 站 内 公告 。 

技术 支持 管理 : 主要 用 于 添加 和 删除 常见 问题 ， 以 及 对 客户 反馈 信息 进行 管理 。 

本 项 目 中 提供 的 后 台 首 页 在 本 书 光盘 中 的 路 径 为 \TM\22\admin\index.php， 打 开 效 果 如 图 22.34 所 示 。 


22.8.2 ”switch 框架 技术 


国共 办 办 办 


网 上 社区 后 台 首 页 的 设计 主要 应 用 switch 语句 和 include 包含 语句 , 其 实现 的 原理 是 : 应 用 switch 语句 ， 


人 
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根据 超 链接 中 传递 的 变量 值 进行 判断 ， 根 据 不 同 的 变量 值 应 用 include 调用 不 同 的 子 文件 。 该 技术 的 实现 流 


程 如 图 22.35 所 示 。 


ws 


图 22.34 BCTY365 网 上 社区 系统 后 台 首 页 


mre so 本 WE 


图 22.35 网 上 社区 后 台 首 页 设计 流程 


为 了 更 好 地 理解 switch 框架 技术 ， 先 来 了 解 一 下 switch 语句 。 该 语句 的 格式 如 下 : 


switch( expr \{ 
case expr1: 


/expr 条 件 为 变量 名 称 
/icase 后 的 expr1 为 变量 的 值 
/冒号 〈:) 后 是 符合 该 条 件 时 要 执行 的 部 分 


/应 用 break 来 跳 离 循环 体 


朋 

参数 expr 是 表达 式 的 值 ， 即 switch 语句 的 条 件 变量 的 名 称 ， 参 数 exprl 放置 于 case 语句 后 ， 是 要 与 条 
件 变量 expr 进行 匹配 的 值 中 的 一 个 ; statementl 是 在 参数 exprl 的 值 与 条 件 变 量 expr 的 值 相 匹配 时 执行 的 代 
码 ，break 语句 终止 语句 的 执行 ， 即 当 语 句 在 执行 过 程 中 ， 遇 到 break 就 停止 执行 ， 跳 出 循环 体 ，default 是 
case 的 一 个 特例 ， 匹 配 了 任何 其 他 case 都 不 匹配 的 情况 ， 并 且 是 最 后 一 条 case 语句 。 

通过 switch 和 include 语句 来 实现 后 台 管理 功能 的 设计 是 一 个 很 好 的 方法 ， 不 但 实现 过 程 简单 ， 而 且 操 
作 也 非常 灵活 。 其 关键 代码 如 下 : 

《代码 位 置 ， 光盘 \TM\Instance\22\admin\wzdh.php) 


<?php 
switch($htgl){ /根据 变量 提交 的 不 同 值 
case "添加 编程 词典 版 本 ": // 判 断 与 变量 提交 的 值 是 否 相 同 
include("addbb.php"); // 如 果 值 相同 ， 则 调用 指定 的 文件 
break; /跳出 本 次 循环 
case "编辑 编程 词典 版 本 ": 
include("editbd.php"); 
break; 
四 /部 分 代码 省 略 
case "": // 当 变量 的 值 为 空 时 
include("edittell.php"); // 调 用 该 文件 
break; 


?> 


@ 
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22.8.3 ”后 台 首 页 的 实现 过 程 


在 后 台 首 页 的 设计 过 程 中 ， 以 switch 循环 语句 为 基础 ， 架 设 整 个 后 台 管理 功能 的 框架 结构 ， 充 分 发 挥 
include 包含 语句 的 作用 , 调用 不 同 的 文件 执行 不 同 的 管理 操作 ; 应 用 JavaScript 脚本 来 控制 栏目 列表 的 输出 

控制 栏目 列表 的 输出 和 隐藏 在 menu.php 文件 中 进行 ， 首 先 定义 一 个 change0 函 数 用 于 控制 表格 的 输出 
和 隐藏 ， 然 后 在 表格 中 应 用 onclick 事件 传递 不 同 的 值 到 自 定义 函数 change0， 最 后 根据 不 同 的 值 显示 不 同 
的 内 容 。 其 关键 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance22\admin\menu.php) 


<script language="javascript"> 
// 通 过 脚本 语言 控制 文本 框 的 伸展 和 收缩 


function change(x,yjf /定义 一 个 函数 
if(x.style.display=="none"}{ // 淹 断 当 样式 的 值 为 none 时 
x.style.display=""; /输出 样式 的 值 为 空 
} 
else if(x.style.display=="™" /| 判断 当 样 式 的 值 为 空 时 
x.style.display="none"; /| 输出 样式 的 值 为 none 
y.background="images/bg_16_11.jpg"; // 输 出 背景 图 片 
中 
} 
</script> 


<table width="175" height="26" border="0" align="center" cellpadding="0" cellspacing="4" 
onclick="change(tz1,img_tz1)" style="cursor:hand"> 
<tr> 


<td background="images/bg_16_11.jpg" id="img_tz1" class="a4"><div align="left"><img 
src="images/bg_16_21.jpg">&nbsp; 编 程 词典 管理 </div></td> 
</tr> 


</table> 
<table name="tz1" id="tz1" width="170" height="40" border="0" align="center" cellpadding="0" cellspacing="0" 
<?php 
// 根 据 变量 的 值 选 择 执行 的 内 容 ， 当 变量 的 值 不 为 真 时 ， 隐 藏 该 表格 
iflisset($_GET[htgl])) 
六 $_GET[htgl]I=" 添 加 编程 词典 版 本 " || $_GET[htgl]I=" 编 辑 编程 词典 版 本 " |l$_GET[htgl]i=" 添 加 编程 词典 " 
ll$_GET[htgl]'= "编辑 编程 词典 " X{ 
?> 


style="display:none" 
<?php 
} 

} 


?> 
> 
<tr> 

<td width="40" height="24" background="images/bg_16_16.jpg">&nbsp;</td> 

<td width="114" background="images/bg_16_16.jpg"><div align="left"><a href="default.php?htgl= 添 加 编程 
词典 版 本 "> 添加 编程 词典 版 本 </a></div></td> 

</tr> 

</table> 


a 
下 培 明 这 里 给 出 的 只 是 后 台 首页 实现 过 程 中 的 主要 代码 ， 详 细 代码 可 参考 本 书 光盘 TMW22\admin 文 


件 夹 下 的 相关 文件 。 
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22.9 ”编程 词典 管理 模块 设计 


狠 i 视频 讲解 : 光盘 \TM\Video\ 第 22 章 \ 编 程 词典 管理 模块 设计 .exe 
本 模块 的 功能 是 对 网 站 中 的 编程 词典 进行 管理 ， 包 括 添加 编程 词典 版 本 、 编 辑 编程 词典 的 版 本 、 添 加 
编程 词典 和 编辑 编程 词典 。 


22.9.1 编程 词典 管理 模块 概述 


本 模块 的 主要 功能 是 管理 网 站 中 在 线 出 售 的 编程 词典 软件 ， 实 现 对 编程 词典 软件 的 更 新 和 维护 ， 其 管 
理 的 内 容 主要 包括 添加 和 编辑 编程 词典 的 版 本 、 添 加 和 编辑 编程 词典 的 详细 信息 。 在 添加 编程 词典 时 ， 包 
括 名 称 、 版 权 、 图 片 、 类 别 、 内 容 简 介 和 不 同 版 本 的 共同 点 ;编辑 编程 词典 包括 版 本 、 价 格 、 简 介 、 功 能 
和 服务 ， 其 中 每 一 个 编程 词典 软件 只 可 以 编辑 一 次 ， 不 可 以 重复 编辑 ， 如 果 要 重新 编辑 ， 就 必须 将 已 经 编 
辑 过 的 信息 删除 。 


22.9.2 图 片上 传 技术 


在 编程 词典 管理 模块 中 ， 应 用 到 图 片上 传 技术 ， 通 过 该 技术 将 编程 词典 的 界面 效果 上 传 到 服务 器 的 指 
定 文件 夹 下 。 该 技术 主要 通过 move_uploaded_ file0) 函 数 来 实现 ， 其 中 还 应 用 到 is_dir0、mkdir0 函 数 ， 判 断 
指定 的 文件 夹 是 否 存在 和 创建 文件 夹 ， 还 有 mt _rand0、strstr0 函 数 和 $_FILES[] 全 局 变量 。 为 了 更 好 地 理解 
和 掌握 图 片上 传 处 理 技术 ， 这 里 以 编程 词典 模块 中 的 savebccd.php 文件 为 例 进行 讲解 。 

首先 应 用 is_dir0 函 数 判断 在 服务 器 中 是 否 存在 指定 的 文件 夹 ， 如 果 不 存在 则 应 用 mkdir0 函 数 创建 一 个 
新 的 文件 夹 。 

然后 应 用 $_FILES[] 全 局 变量 获取 图 片 名 ， 应 用 strstr0 函 数 获 取 图 片 文件 的 后 级 名 ， 为 避免 出 现 同名 文 
件 覆 盖 ， 这 里 应 用 系统 的 当前 时 间 和 mtrandO 函 数 获取 的 一 个 7 位 随机 数字 作为 图 片 的 名 称 。 

最 后 确定 图 片 在 服务 器 中 存储 的 路 径 ， 将 图 片上 传 到 指定 的 文件 夹 下 。 而 数据 库 中 存储 的 数据 是 图 片 在 
服务 器 中 的 路 径 ， 当 需要 输出 图 片 时 ,只 需要 获取 数据 库 中 图 片 的 路 径 即 可 。savebccd.php 文件 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\22\admin\savebccd.php) 

<?php include_once("../conn/conn.php"); /连接 数据 库 

$bccdname=$_POST[bccdname]; /获取 POST 方法 提交 的 值 

$owner=$_POST[owner]; 

$typeid=$_POST[typeid]; 

$content=$_POST[T'content]; 

$samepart=$_POST['samepart]; 


$addtime=date("Y-m-j Hii:s"); /获取 当前 时 间 

@ ifis_dir"./bccdimages")==falseX{ // 潮 断 指定 的 文件 是 否 存 在 

@ mkdir("/bccdimages"); // 如 果 不 存在 ， 则 创建 一 个 新 的 文件 夹 
S$link=date("YmjHis"); // 获 取 当 前 时 间 


// 为 表单 中 提交 的 数据 重新 命名 ， 以 当前 时 间 和 随机 数 作为 名 称 

// 其 中 使 用 $_FILES 获取 表单 中 真实 的 名 称 ， 使 用 strstr 函数 获取 文件 的 后 组 

© $path=$link.mt_rand(1000000,9999999).strstr($_FILES["imageaddress"]["name"],"."); 
$address="./bccdimages/".$path; /定义 文件 上 传 的 路 径 

@ move_uploaded_file($_FILES["imageaddress"]["tmp_name"],$address);// 将 文件 上 传 到 指定 的 文件 中 


@ 
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Simageaddress="./admin/bccdimages/".$path; // 获 取 上 传 文件 在 服务 器 中 的 存储 路 径 


// 将 表单 中 提交 的 数据 存储 到 数据 库 中 
$query=mysql_query("insert into tb_bccd(bccdname,owner,typeid,content,samepart,imageaddress,addtime) 
values('$bccdname','$owner','$typeid",'$content','$samepart','$imageaddress','$addtime'")",$conn); 
if($query==trueX{ 

echo "<script>alert(' 编 程 词典 添加 成 功 ! ');history.back();</script>"; 
Jelse{ 

echo "<script>alert( 编程 词典 添加 失败 ! ');history.back();</script>"; 


?> 


~ 
说明 @is dir0: 判断 指定 的 文件 夹 是 否 存 在 ， 如 果 存 在 则 返回 true， 否 则 返回 false。 

@ mkdir0: 创建 一 个 新 的 文件 夹 。 

@ mt_rand(): 根据 提供 的 参数 min 和 max 生成 随机 数 ， 如 果 没 有 提供 可 选 参数 min 和 max， 则 返回 
0~RAND MAX 之 间 的 伪 随 机 数 。 

strstr0): 获取 一 个 指定 字符 串 在 另 一 个 字符 串 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字符 囊 。 如 果 执 行 
成 功 ， 则 返回 剩余 字符 囊 ( 存在 相 匹 配 的 字符 ); 如 果 没 有 找到 相 匹 配 的 字符 ， 则 返回 false。 

$_FILES[]: 全 局 变量 ， 获 取 所 有 上 传 文件 的 信息 。 该 全 局 变量 还 可 以 获取 到 其 他 的 值 ， 其 中 
$_FILES['imageaddress'][mame'] 获 取 的 是 客户 端 机 器 文件 的 原名 称 ; $ FILES['imageaddress']['size'] 获 取 已 
上 传 文件 的 大 小 , 单位 为 字 节 ; $_FILES['imageaddress']['tmp_name'] 获 取 文 件 被 上 传 后 在 服务 端 储存 的 临 
时 文件 名 ; $_FILES['imageaddress']['error] 获 取 和 该 文件 上 传 相关 的 错误 代码 。 

@ move_ uploaded file0: 应 用 POST 方法 实现 文件 的 上 传 ， 参 数 filename 指定 要 上 传 的 文件 地 址 ; 
参数 destination 指定 文件 上 传 到 服务 器 后 的 存储 目录 及 名 称 。 


sa 
"multipart/form-data" 。 


22.9.3 ”添加 编程 词典 模块 的 实现 过 程 


添加 编程 词典 模块 的 功能 是 向 数据 库 中 添加 编程 词典 的 详细 信息 ， 包 括 编程 词典 的 名 称 、 版 权 、 图 片 、 
类 别 、 内 容 简 介 和 不 同 版 本 的 共同 点 。 其 运行 效果 如 图 22.36 所 示 。 


[3 
EE 


EE EE 
5 可 


| 


和 内科 介 ; 
| 
| 


FARFERS 


习 


ET 
图 22.36 添加 编程 词典 模块 的 运行 效果 
添加 编程 词典 模块 主要 通过 addbccd.php 和 savebccd.php 文件 来 完成 ， 其 中 addbccd.php 文件 主要 用 于 


@ 
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设计 表单 元 素 ， 而 savebccd.php 文件 主要 是 对 表单 中 提交 的 数据 进行 处 理 。addbccd.php 文件 中 使 用 的 表单 
元 素 如 表 22.5 所 示 。 


表 22.5 ”添加 编程 词典 页 中 使 用 的 重要 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含义 
二 jo Method “pos action="savebccd.php"” onSubmit="returm chkinput(this)" 编程 词典 表单 
enctype="multipart/form-data”> 
bccdname text size="25" class="txt grey" 编程 词典 名 称 
Owner text " size="25" class="txt grey" 版 权 所 有 者 
imageaddress | _ file Size="25" class="txt grey" 界面 图 片 
<?php include once("../conn/conn.php"): 
$sql=mysql_query("select * from tb_type order by createtime desc".$conn): 
Sinfo=mysql fetch_array($sqD):; 
if($info 一 false){ echo "<option > 暂 无 类 别 </option>"; 选择 编程 词典 的 
typeid select 
jelsef 版 本 
dof 
echo "<option value=".$infof'id].">".$info['typename']."</option>": } 
While($info=mysql fetch array($sqD):} ?> 
content textarea |rows="10" cols="65" class="textarea"> 内 容 简 介 
samepart textarea |rows="10" cols="65" class="textarea"> 不 同 版 本 的 特点 
Submit submit Value=" 添 加 " class="btn_grey” 提交 表单 


savebccd.php 文件 实现 对 表单 中 提交 的 数据 进行 处 理 ， 首 先 通过 $_POST 获取 表单 中 提交 的 数据 ， 然 后 

判断 指定 的 文件 夹 是 否 存在 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 其 关键 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\22\admin\savebccd.php) 

<?php include_once("../conn/conn.php"); /| 连接 数据 库 

$bccdname=$_POST[bccdname']; /获取 POST 方法 提交 的 值 

$owner=$_POST[owner]; 

$typeid=$_POST[typeid]; 

$content=$_POST[content]; 

$samepart=$_POST[samepart]; 


$addtime=date("Y-m-j H:i:s"); // 获 取 当 前 时 间 

@ if(lis_dir("./bccdimages")==falseX{ // 淹 断 指定 的 文件 是 否 存 在 

@ mkdir("/bccdimages"); // 如 果 不 存 在 ， 则 创建 一 个 新 的 文件 夹 
Slink=date("YmjHis"); // 获 取 当 前 时 间 

eben 以 当前 时 间 和 随机 数 作为 名 称 ， 其 中 使 用 $_FILES 获取 表单 中 真实 的 名 称 ， 使 用 
strstr() 函 数 获取 文件 的 后 

© $path=$link.mt ee ,9999999).strstr($_FILES["imageaddress"]["name"],"."); 
$address="./bccdimages/".$path; /定义 文件 上 传 的 路 径 
move_uploaded file($_FILES["imageaddress"] [tmp_name"],$address); /将 文件 上 传 到 指定 的 文件 中 
S$imageaddress="./admin/bccdimages/".$path; // 获 取 上 传 文件 在 服务 器 中 的 存储 路 径 
// 将 表单 中 提交 的 数据 存储 到 数据 库 中 


$query=mysql_query("insert into tb_bccd(bccdname,owner,typeid,content,samepart,imageaddress,addtime) 
values('$bccdname','$owner','$typeid','$content','$samepart','$imageaddress','$addtime')", $conn); 
if($query==trueX{ 

echo "<script>alert(' 编 程 词典 添加 成 功 !");history.back();</script>"; 
}else{ 

echo "<script>alert(' 编 程 词典 添加 失败 !");history.back();</script>"; 
} 


?> 
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/ 
SO 培 明 @is dir0: 判断 指定 的 文件 是 否 存在 。 

@ mkdir0: 创建 一 个 新 的 文件 夹 。 

mt rand0: 获取 随机 数字 。 

strstr0: 获取 一 个 指定 字符 串 在 另 一 个 字符 串 中 首次 出 现 的 位 置 到 后 者 末尾 的 子 字符 串 。 如 果 执行 
成 功 ， 则 返回 剩余 字符 囊 ( 存在 相 匹 配 的 字符 ) ; 如 果 没有 找到 相 匹配 的 字符 ， 则 返回 false。 


22.9.4 ”编辑 编程 词典 模块 的 实现 过 程 


在 完成 对 编程 词典 信息 的 添加 后 ， 接 下 来 就 可 以 对 编程 词典 的 版 本 信息 进行 编辑 ， 主 要 添加 版 本 信息 、 
价格 、 简 介 、 功 能 和 推出 的 服务 。 该 模块 的 运行 效果 如 图 22.37 所 示 。 
| 


| 可 
ee 


E 到 要 | 
图 22.37 编辑 编程 词典 模块 的 运行 效果 


该 功能 的 实现 同样 通过 两 个 文件 ， 一 个 是 提交 表单 的 editbccd.php 文件 ， 一 个 是 处 理 表 单 提交 数据 的 
sacvbccdbb.php 文件 。 提 交 表单 文件 editbccd.php 中 使 用 的 表单 元 素 如 表 22.6 所 示 。 


表 22.6 编辑 编程 词典 页 中 使 用 的 重要 表单 元 素 


重要 属性 含义 
2 让 00 action="savebccdbb.php" onSubmit="return 编辑 编程 词典 表单 
chkinput(this)"> 
Class="txt_grey" disabled="disabled" value=" 
<?php 
$sql4=mysql_query("select bccdname from tb_bccd where 编程 词典 名 称 , 这 里 设置 了 文 
id=-".$ GET[bccdid]."".$Sconm): 本 的 只 读 属性 

Sinfo4=mysql fetch array($sql4): 
echo unhtml($info4['bccdname'D): ?> 

"<?php echo $ GET[bccdid]:?> 


forml 


bccdname text 
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称 | 元 素 类 型 重要 属性 含义 


bbid 


<?php $sql3=mysql query("select * from tb bb order by 
createtime desc ".$conn): 

Sinfo3=mysql fetch array($sql3); 

if($info3 一 false){ echo "<option> 暂 无 版 本 信息 </option>"; 


select j}else{ 选择 编程 词典 的 版 本 
dof > 
<option value="<?php echo $info3['id]:?>"><?php echo 
Unhtml($info3[bbname']):?></option> 
<?php }while($info3=mysql fetch array($: 


Submit submit Value=" 添 加 " class="btn grey" 提交 表单 


否 已 
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sacvbccdbb.php 文件 对 表单 提交 的 数据 进行 处 理 ， 首 先 获取 表单 中 提交 的 数据 ， 然 后 判断 指定 的 版 本 是 


经 被 添加 ， 最 后 将 数据 存储 到 指定 的 数据 表 中 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\22\admin\savebccdbb.php) 

<?php 

$bccdid=$_POST[bccdid]; // 获 取 表 单 中 提交 的 数据 
$bbid=$_POST[bbid]; // 获 取 编程 词典 id 
$price=$_POST[price]; // 获 取 编 程 词典 单价 
$content=$_POST[content]; /获取 编程 词典 内 容 
$gn=$_POST[gn1; // 获 取 编 程 词典 功能 
$fw=$_POST[fw]; // 获 取 编 程 词典 服务 
include_once("../conn/conn.php"); /| 连接 数据 库 文件 


// 判 断 提 交 的 编程 词典 是 否 已 经 被 添加 
$sql=mysql_query("select id from tb_bbqb where bccdid=".$bccdid."",$conn); 


$info=mysql_fetch_array($sql); /检索 指定 编程 词典 的 id 
if($info!=false}{ // 如 果 检索 值 为 真 ， 则 弹出 提示 
echo "<script>alert(' 该 版 编程 词典 已 经 添加 !");history.back();</script>"; 
exit; 


/和 

$query=mysql_query("insert into tb_bbqb(bccdid,bbid,price,content,gn,fw) 

values('$bccdid','$bbid','$price','$content','$gn','$fw')", $conn); // 将 表单 中 提交 的 数据 存储 到 数据 库 中 

// 更 新 编程 词典 的 价格 

$querys=mysql_query("update tb_bccd set bbid='$bbid ,price='$price' where id=".$bccdid.”"); 

if($query==true and $querys==trueX{ // 如 果 添 加 和 更 新 操作 为 真 ， 则 弹出 提示 
echo "<script>alert(' 版 本 信息 添加 成 功 !");history.back();</script>"; 

}else{ // 如 果 添 加 和 更 新 操作 为 假 ， 则 弹出 提示 
echo "<script>alert(' 版 本 信息 添加 失败 !");history.back();</script>"; 

上 


?> 


22.10 软件 升级 管理 模块 设计 


艾 视频 讲解 : 光盘 \TM\Video\ 第 22 章 \ 软 件 升级 管理 模块 设计 .exe 
10.1 软件 升级 管理 模块 概述 


软件 升级 管理 模块 实现 对 软件 升级 包 的 管理 ， 其 具体 的 功能 包括 添加 升级 包 、 编 辑 升级 包 、 添 加 序列 
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号 和 编辑 序列 号 。 软 件 升级 管理 模块 中 的 添加 升级 包 和 添加 序列 号 是 一 一 对 应 的 ， 其 中 根据 所 属 的 类 别 和 
版 本 来 确定 升级 包 对 应 的 序列 号 ， 每 一 个 版 本 一 个 类 别 的 升级 包 对 应 一 个 序列 号 。 


22.10.2 ”动态 输出 下 拉 列 表 框 的 值 


在 软件 升级 管理 模块 中 ， 应 用 到 一 个 动态 输出 下 拉 列 表 框 中 值 的 技术 。 下 面 就 来 讲解 一 下 该 技术 是 如 
何 实现 的 。 
在 讲解 该 技术 之 前 ， 先 来 了 解 一 下 下 拉 列 表 框 的 基本 结构 。 
<select name="select"><!-->name 指定 该 下 拉 列 表 框 的 名 称 <!--> 
<!-->selected 设置 下 拉 列 表 框 的 默认 值 ， 默 认 值 为 PHP<!-> 
<option selected="selected">PHP</option> 
<!->value 指定 的 "mysql" 是 下 拉 列 表 框 传递 的 值 ，"MYSQL 为 显示 的 内 容 <!-> 
<option value="mysql">MYSQL</option> 
</select> 
所 谓 动态 输出 下 拉 列 表 框 中 的 值 就 是 从 数据 库 中 读 取 数 据 ， 将 获取 到 的 数据 输出 到 下 拉 列 表 框 中 ， 而 
不 是 直接 在 下 拉 列 表 框 中 设置 某 个 固定 的 值 . 这 里 以 软件 升级 管理 模块 文件 addsjb.php 中 的 所 属 类 别 下 拉 列 
表 框 为 例 进 行 讲解 ， 其 中 设置 下 拉 列 表 框 的 名 称 为 typeid， 默 认 值 为 “请 选择 ”，value 的 值 是 从 数据 库 中 
获取 的 ID 值 ， 显 示 的 内 容 为 从 数据 库 中 获取 的 类 型 名 称 。 动 态 输出 下 拉 列 表 框 中 值 使 用 的 关键 代码 如 下 : 
《代码 位 置 ， 光盘 \TMNInstance\22\adminvwzdh-php) 
< 一 一 一 一 一 一 一 一 一 设置 下 拉 列 表 框 的 名 称 为 typeid- 一 一 一 一 一 一 一 -> 
<select name="typeid" class="txt_grey"> 
<option value=" selected="selected"> 请 选择 </option> 
<?php 
include_once("../conn/conn.php"); /| 连接 数据 库 
// 从 数据 库 中 读 取 编 程 词典 类 型 的 数据 
$sql=mysql_query("select * from tb_type order by createtime desc",$conn); 
$info=mysql_fetch_array($sql); 
if($info==false){ 
echo "<option > 暂 无 类 别 </option>"; 


jelsef 
dof /应 用 do…while 循环 语句 输出 类 型 的 ID 和 名 称 
echo "<option value=".$info['id].">".$info[typename]."</option>"; 
while($info=mysql_fetch_array($sql)); /do…while 循环 语句 结束 
?> 
</select> 


下 拉 列 表 框 不 但 可 以 动态 输出 数据 库 中 某 个 字段 的 数据 ， 而 且 还 可 以 输出 数组 中 的 数据 。 下 面 就 实现 
一 个 在 下 拉 列 表 框 中 动态 输出 数组 中 数据 的 功能 ， 首 先 创建 一 个 下 拉 列 表 框 ， 然 后 设置 下 拉 列 表 框 的 值 ， 
从 数组 中 读 取 数据 ， 应 用 for 循环 语句 进行 输出 。 其 代码 如 下 : 


< 一 一 一 一 一 一 一 一 一 设置 下 拉 列 表 框 的 名 称 为 select 一 一 一 一 一 一 一 一 一 一 一 
<select name="select" size="1"> 
<?php 
$string="ASP,PHP,JSP,.NET,DEL,VB,VC"; /定义 一 个 字符 串 
$srtings=split(",",$string); /分 隔 字符 串 
$count=count($srtings); /获取 数 组 中 元 素 的 数量 
for($i=0:$i<$count$i++f /根据 数组 中 元 素 的 数量 进行 循环 输出 
$result=$srtings[$i]; /定义 变量 ， 获 取 数 组 中 指定 的 元 素 
echo "<option>S$result</option>";} /将 数组 中 的 元 素 输出 到 下 拉 列 表 框 中 
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?> 
</select> 
动态 输出 数据 库 中 数据 到 下 拉 列 表 框 的 运行 效果 如 图 22.38 所 示 。 


10.3 ”软件 升级 包 上 传 的 实现 过 程 


软件 升级 包 上 传 在 添加 升级 包 模块 中 实现 ， 通 过 一 个 文件 域 文 本 框 将 升级 包 提交 到 服务 器 中 指定 的 文 


件 下 ， 并 且 将 该 文件 在 服务 器 中 的 路 径 存储 到 数据 库 中 ， 便 于 在 前 台 实现 对 软件 升级 包 的 下 载 。 其 运行 效 


果 如 图 22.39 所 示 。 
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22.38 ”动态 输出 数据 库 中 数据 到 下 拉 列 表 框 22.39 ”软件 升级 包 上 传 的 运行 效果 


理 。 


在 本 模块 中 通过 addsjb.php 文件 来 提交 升级 包 的 信息 ， 通 过 savesj.php 文件 来 对 表单 提交 的 数据 进行 处 
其 中 在 将 升级 包 上 传 到 服务 器 的 指定 文件 夹 的 过 程 中 ， 主 要 应 用 的 是 move_uploaded file0 函 数 。 在 


savesj.php 文件 中 ， 首 先 获 取 表 单 提交 的 数据 ， 然 后 判断 服务 器 中 是 否 存在 指定 的 文件 ， 最 后 应 用 
move_uUploaded file0 函 数 将 升级 包 上 传 到 指定 的 文件 夹 下 ,并 且 将 数据 存储 到 指定 的 数据 表 中 。 其 程序 代码 
如 下 : 
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(代码 位 置 ， 光 盘 \TM\Instance\22\admin\savesj.php) 

<?php 
Sname=$_POST[name']:; // 获 取 表 单 提交 的 数据 
S$typeid=$_POSTT'typeid']: // 获 取 表单 提交 的 数据 
$content=$_POST[content]; // 获 取 表 单 提 交 的 数据 
$addtime=date("Y-m-j H:i:s"); /定义 时 间 变 量 
$bbid=$_POST[bbid']; // 获 取 表 单 提交 的 数据 
iflis_dir("./sjxz")==falseX{ /| 判断 指定 的 文件 夹 是 否 存在 

mkdir("./sjxz"); // 如 果 指 定 的 文件 夹 不 存在 ， 则 创建 一 个 指定 的 文件 夹 
} 
Slink=date("YmjHis"); // 获 取 一 个 时 间 
$path=$link.mt_rand(1000000,9999999).strstr($_FILES["address"]["name'"],""); /重新 设置 升级 包 名 称 
$address="./sjxz/".$path; // 设 置 升级 包 在 服务 器 中 存储 的 指定 路 径 
move_uploaded_file($_FILES["address"]["tmp_name"],$address); /将 升级 包 上 传 到 指定 的 路 径 下 
$address="./admin/sjxz/". $path; // 获 取 升 级 包 在 服务 器 中 的 存储 路 径 
include_once("../conn/conn.php"); // 连 接 数 据 库 文 件 


// 将 上 传 的 数据 存储 到 数据 库 中 ， 这 里 将 升级 包 在 服务 器 中 的 路 径 存储 到 数据 库 中 
$query=mysql_query("insert into tb_sjxz(name,typeid,content,addtime,address,bbid) 
values('$name','$typeid','$content','$addtime','$address','$bbid')", $conn); 


if($query\{ // 如 果 添加 操作 成 功 ， 则 弹出 提示 
echo "<script>alert( 升 级 包 添加 成 功 !");history.back();</script>"; 
}else{ // 如 果 添加 操作 失败 ， 则 弹出 提示 


echo "<script>alert( 升 级 包 添加 失败 !");history.back();</script>"; 
} 


?> 
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22.10.4 软件 升级 包 删 除 的 实现 过 程 


软件 升级 包 删 除 的 实现 主要 根据 当前 数据 中 提供 的 ID,， 执行 delete 删除 语句 ， 将 数据 表 中 相同 ID 的 数 
据 删 除 。 其 运行 效果 如 图 22.40 所 示 。 
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图 22.40 软件 升级 包 删 除 的 运行 效果 
该 功能 的 实现 主要 通过 editsjb.php 和 deletesjb.php 文件 。 通 过 editsjb.php 文件 输出 数据 库 中 存储 的 有 关 
升级 包 的 信息 ， 以 分 页 的 形式 显示 ， 在 每 条 记录 的 最 后 设置 一 个 删除 链接 ， 通 过 脚本 来 调用 deletesjb.php 文 
件 ， 根 据 变量 中 的 ID 值 执行 删 除 升级 包 的 操作 。 其 关键 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\22\admin\deletesjb.php) 


<?php 
$id=$_GET[id]; /获取 变量 传递 的 ID 
include_once("../conn/conn.php"); /| 连接 数据 库 


// 执 行 删除 操作 ， 将 数据 表 中 对 应 的 ID 的 数据 删除 
if(mysql_query("delete from tb_sjxz where id=".$id."",$conn)){ 
echo "<script>alert(' 该 升级 包 删 除 成 功 ");history.back();</script>";”// 如 果 删除 操 作成 功 ， 则 弹出 提示 
}else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 
echo "<script>alert(' 该 升级 包 删 除 失败 人);history.back();</script>"; 
} 


?> 


22.11 在 线 支付 技术 专题 


所 谓 在 线 支付 就 是 客户 端 〈 金 融 机 构 需 客户 端 安装 由 金融 机 构 签发 的 数字 证 书 ， 信 用 卡 免 安 装 ) 将 支 
付 信息 加 密 后 通过 互联 网 传送 到 支付 网 关 〈 支 付 网 关 是 解决 网 络 上 安全 支付 问题 的 交易 平台 ， 位 于 互联 网 
和 传统 的 金融 机 构 内 部 网 之 间 ， 其 主要 作用 是 将 互联 网 和 金融 网 络 安全 地 连接 起 来 ， 将 不 安全 的 网 上 交易 
信息 传 给 安全 的 金融 网 络 ， 起 到 隔离 和 保护 金融 网 络 的 作用 ) ， 同 时 金融 机 构 网 上 支付 系统 反馈 有 关 支 付 
信息 ， 客 户 确认 无 误 后 进行 支付 确定 ， 支 付 网 关 负责 商户 网 上 交易 资金 的 清算 ， 并 根据 商户 提供 的 开户 行 、 


全 
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账号 等 结账 信息 将 网 上 消费 款项 汇总 划 入 商户 账户 。 


BCTY365 网 上 社区 的 在 线 支付 是 与 中 国 工商 银行 合作 来 完成 的 BCTY365 网 上 社区 的 在 线 支付 操作 步 
又 如 下 : 


(1) 登录 网 上 社区 ， 如 图 22.41 所 示 。 


(2) 购买 商品 。 在 本 页 中 ， 不 但 可 以 购买 商品 ， 还 可 以 查看 商品 的 详细 信息 和 购物 车 中 的 商品 信息 ， 
2.42 所 示 。 


共有 册 可 司 虹 2 个 二 页 旺 元 8 个 站 了 页 / 共 工 页 机 上 -页 下- 页 局 页 


22.41 在 线 订购 的 操作 页 面 


(3) 进入 购物 车 操作 页 面 ， 如 图 22.43 所 示 。 在 该 页 面 中 ， 可 以 修改 购物 数量 、 删 除 指定 商品 、 清 空 
购物 车 、 继 续 购物 和 统计 购买 商品 的 金额 ， 也 可 以 单 击 “ 结 算 ” 按 钮 进入 到 商品 结算 页 面 。 
(4) 进入 到 购物 结算 页 面 ， 填 写 收 货 人 的 详细 信息 ， 确 认 后 提交 该 数据 ， 如 图 22.44 所 示 。 
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图 22.43 ”购物 车 操作 页 面 图 22.44 ”填写 收 货 人 的 详细 信息 


(5) 订单 确认 。 订 单 确认 以 后 ， 就 可 以 提交 订单 ， 准 备 进行 网 上 支付 ， 如 图 22.45 所 示 。 
(6) 进行 网 上 支付 。 在 这 里 可 以 选择 工行 网 上 支付 ， 也 可 以 选择 取消 该 订单 ， 如 图 22.46 所 示 。 
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图 22.45 订单 确认 图 22.46 执行 网 上 支付 


第 22 章 “综合 实例 (四 ) 一 一 BCTY365 网 上 社区 


接 下 来 的 操作 在 工行 B2C 支付 页 面 上 进行 。 首先 网 上 社区 按照 工商 银行 B2C 订单 数据 规范 形成 提交 数 
据 ， 并 使 用 工商 银行 提供 的 API 和 商户 证 书 对 订单 数据 签名 ， 形 成 form 表单 返回 客户 浏览 器 ， 表 单 action 
地 址 指向 工商 银行 接收 商户 B2C 订单 信息 的 servlet， 然 后 在 客户 确认 使 用 工行 网 上 支付 后 ， 提 交 此 表单 到 
工商 银行 ， 最 后 工行 网 银 系统 接收 此 笔 B2C 订单 ， 对 订单 信息 和 商户 信息 进行 检查 ， 通 过 检查 则 显示 工行 
B2C 支付 页 面 。 

客户 通过 工行 B2C 支付 页 面 实现 网 上 支付 ， 商 户 查 询 网 上 银行 的 账户 ， 如 果 货 款 已 经 到 账 ， 则 根据 客 
户 指定 的 方式 将 货物 送 达 客户 手中 。 

上 述 内 容 就 是 网 上 社区 系统 的 在 线 支付 流程 ， 涉 及 到 工商 银行 的 操作 内 容 这 里 不 做 讲解 。 这 里 主要 讲 
解 一 下 如 何 将 订单 信息 提交 到 工商 银行 。 该 项 操作 主要 通过 shopping_tjdd.php 文件 来 实现 , 首先 从 数据 库 中 
读 取 订单 信息 , 然后 将 订单 信息 输出 , 最 后 创建 “取消 订购 ”和 “工行 网 上 支付 ”两 个 超 链接 , 通过 JavaScript 
脚本 来 调用 不 同 的 执行 文件 。 其 关键 代码 如 下 : 

<?php include_once("conn/conn.php"); include_once("top.php"); ?> ”// 连 接 数 据 库 和 网 站 的 头 文件 

<!-> 省 略 了 部 分 代码 <!-> 

<?php 

@ $ddnumber=base64_decode($_GET["ddno"); // 对 获取 的 订单 编号 进行 base64 解码 

/获取 该 订单 的 金额 信息 

$sql=mysql_query("select * from tb_dd where ddnumber=".$ddnumber."",$conn); 

$info=mysql_fetch_array($sql); 


$amount=$info["totalprice"]; 

@ $amount=str_replace(",","",number_format($amount,2)); /修改 数字 的 输出 格式 
$amount=str_replace("",",number_format($amount,2)); /修改 数字 的 输出 格式 
?> 


Ee // 省 略 部 分 HTML 代码 
<table width="630" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr><td width="159">&nbsp; 
<?php 
$sql=mysql_query("select totalprice from tb_dd where ddnumber=".base64_decode($_GET["ddno"])."",$conn); 
$info=mysql_fetch_array($sql); 
echo "<font color=red><strong>".$info["totalprice"]."&nbsp; 元 </strong></font>"; 
?> </td> 
</tr> 
</table> 
a // 省 略 部 分 HTML 代码 
<script language="javascript"> 
/打印 订单 
©@ function openprintwindow(x,y,zX{ 
window.open("printwindow.php?ddno="+x+"&pv="+z,"newframe","top=200,left=200,width=635,height="+( 
230+20*y)+",menubar=no,location=no,toolbar=no,scrollbars=no,status=no"); 


! 
</script> 
as // 省 略 部 分 HTML 代码 
<table width="630" height="25" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr> 
< 取消 该 订单 三 


<td width="75"><img src="images/bg_14(14).jpg" width="69" height="20" style="cursor:hand" 
onclick="javascript:if(window.confirm(' 如 果 取 消 该 订单 ， 则 该 订单 将 被 删除 ， 您 需要 重新 购买 ! 
人 location.href='deletedd.php?ddno=<?php echo $_GET["ddno"];?>";}"/></td> 
执行 网 上 支付 = 
ee width="125"><img src="images/bg_14(15).jpg" width="119" height="20" 
onclick="javascript:window.location.href='ddform.php?orderid=<?php echo 
base64_decode($_GET["ddno"]);?>&amount=<?php echo $amount;?>&orderDate=<?php echo date("Ymdhis");?>";" 
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style="cursor:hand"/></td> 


</table> 


< 省 略 了 部 分 代码 站 
<?php include_once("bottom.php"); ?> // 包 含 网 站 的 尾 文件 


/ 
后 明 @ base64 decode0: PHP 实现 对 base64 编码 的 字符 进行 解码 。PHP 实现 字符 囊 的 base64 编码 

通过 base64_encode0) 函 数 。 

@ str replace(0: 实现 字符 串 的 替换 。 该 函数 的 语法 如 下 : 

mixed str_replace ( mixed search, mixed replace, mixed subject , int &count ) 

str_replace() 将 所 有 在 参数 subject 中 出 现 的 search 以 参数 replace 替换 ， 参 数 &count 表示 替换 字符 串 
执行 的 次 数 。 

目 openprintwindow0: JavaScript 脚本 中 自 定义 的 函数 ， 用 于 执行 订单 的 打印 操作 。 


22.12 小 结 


本 章 从 项 目 开发 的 实际 角度 出 发 ， 详 细 地 讲解 了 BCTY365 网 上 社区 系统 的 开发 过 程 ， 其 中 以 系统 的 整 
体 开 发 流程 为 主线 ， 重 点 介绍 技术 支持 、 在 线 订 购 、 社 区 论坛 和 编程 词典 等 几 个 大 模块 的 实现 方法 ， 并 且 
对 管理 员 权限 设置 、 帖 子 置顶 设置 和 在 线 支付 技术 进行 讲解 。 
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( 铝 / 视频 讲解 : 70 分 钟 ) 


目前 网 络 上 针对 PHP 的 模板 数不胜数 。 作 为 最 早 的 MVC 模板 之 
一 ，Smarty 在 功能 和 速度 上 处 于 绝对 的 领先 地 位 。 那 么 ，Smarty 的 将 
点 是 什么 ? 它 是 如 何 完成 代码 分 离 的 ? 

通过 阅读 本 章 内 容 ， 你 可 以 : 


是 
[ed 
Pl 
Lad 
Ph 
Lad 


了 解 什么 是 MVC， 什 么 是 Smarty 
掌握 Smarty 的 特点 

掌握 Smarty 模板 的 安装 和 配置 的 方法 
掌握 Smarty 模板 设计 的 方法 

掌握 Smarty 程序 设计 的 方法 

熟悉 Smarty 模板 的 应 用 


第 23 章 Smarty 模板 技术 


23.1 Smarty 简介 


Smarty 是 PHP 中 的 一 个 模板 引擎 ， 是 众多 PHP 模板 中 最 优秀 、 最 著名 的 模板 之 一 。Smarty 模板 引擎 将 
PHP 程序 直接 生成 模板 文件 , 最 终 浏览 器 中 读 取 的 就 是 Smarty 模板 文件 ， 并 且 Smarty 能 够 对 模板 文件 进行 
判断 ， 如 果 是 第 一 次 或 者 模板 已 经 改变 ， 则 重新 生成 模板 文件 ， 否 则 将 直接 执行 原 模板 文件 。Smarty 模板 


引擎 的 运行 流程 如 图 23.1 所 示 。 
如 果 是 第 一 
次 或 模板 已 
经 改变 ， 重 


解析 模板 
读 取 模 板 文件 变量 内 容 


PHP 程序 

| | = 
i 执行 | 

| | 如 果 不 是 第 

己 编 译 的 一 次 或 者 模 

| 和 | 板 未 改变 则 

I 使 用 原文 件 
.| | 


图 23.1 Smarty 模板 引擎 运行 流程 图 
23.1.1 Smarty 模板 引擎 


当 动态 网 页 开始 风行 时 ， 网 站 设计 师 们 对 HTML 中 嵌入 脚本 语言 又 提出 了 更 高 的 要 求 ， 对 这 种 方式 很 
不 满意 。 因 为 无 论 是 微软 的 ASP 还 是 开放 源码 的 PHP， 都 属于 服务 器 端的 HTML 嵌入 式 脚本 语言 ， 作 为 一 
个 网 站 设计 师 必 须 既 懂 程序 设计 ， 又 懂 页 面 设计 。 但 在 通常 情况 下 ， 如 果 程序 设计 是 高 手 ， 那 么 在 页 面 设 
计 上 必然 差 一 些 ， 反 之 也 是 如 此 ， 正 所 谓 “ 鱼 和 熊 掌 不 可 兼 得 ”。 由 此 有 网 站 设计 师 设想 ， 如 果 能 将 网 页 
设计 与 程序 开发 分 离 ， 效 果 是 否 会 更 好 呢 ? 因此 模板 引擎 应 运 而 生 。 

开发 模板 引擎 的 目的 是 实现 程序 设计 与 网 页 设计 的 逻辑 分 离 ， 使 程序 设计 者 可 以 专注 于 程序 功能 的 开 
放 ; 而 网 页 设计 师 专注 于 页 面 的 设计 ， 让 网 页 看 起 来 更 加 具有 专业 性 。 

Smarty 是 一 个 使 用 PHP 编写 的 应 用 于 PHP 的 模板 引擎 ， 它 将 一 个 应 用 程序 分 成 两 部 分 : 视图 和 逻辑 控 
制 。 简 单 地 讲 ， 就 是 将 UI〈 用 户 界面 ) 和 PHP code (PHP 代码 ) 分 离 。 这 样 ， 程 序 员 在 修改 程序 时 不 会 影 
响 到 页 面 设计 ， 而 美工 在 重新 设计 或 修改 页 面 时 也 不 会 影响 程序 逻辑 。 


人 
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23.1.2 ”Smarty 与 MVC 


Smarty 的 这 种 开发 模式 是 基于 MVC 框架 的 概念 。MVC (Model-View-Controller) ， 即 模型 -视图 -控制 
器 。MVC 框架 将 一 个 应 用 程序 定义 成 视图 、 控 制 器 、 模 型 3 部 分 。 

模型 : 对 接收 过 来 的 信息 进行 处 理 ， 并 将 处 理 结果 回 传 给 视图 。 例 如 ， 如 果 用 户 输入 信息 正确 ， 

那么 将 给 视图 一 个 命令 ， 允 许 用 户 进入 主页 面 ; 反之 ， 则 拒绝 用 户 的 操作 。 

视图 : 就 是 提供 给 用 户 的 界面 。 视 图 只 提供 信息 的 收集 及 显示 ， 不 涉及 处 理 。 如 用 户 登录 ， 登 录 
界面 就 是 视图 ， 只 提供 用 户 登录 的 用 户 名 和 密码 输入 框 〈 也 可 以 有 验证 码 、 安 全 问题 等 信息 ) ， 
至 于 用 户 名 和 密码 的 对 与 错 ， 这 里 不 去 处 理 ， 直 接 传 给 后 面 的 控制 部 分 。 
控制 器 : 负责 处 理 视图 和 模型 的 对 应 关系 ， 并 将 视图 收集 的 信息 传递 给 对 应 的 模型 。 例 如 ， 当 用 
户 输入 用 户 名 和 密码 后 提交 。 这 时 ， 控 制 部 分 接收 用 户 的 提交 信息 ， 并 判断 这 是 一 个 登录 操作 ， 
随后 将 提交 信息 转发 给 登录 模块 部 分 ， 也 就 是 模型 。 


[sl 


23.1.3 Smarty 特点 


采用 Smarty 模板 编写 的 程序 可 以 获得 最 快 的 速度 。 注 意 ， 这 是 相对 于 其 他 模板 而 言 。 
可 以 自行 设置 模板 定 界 符 ， 如 他、{f}、<!-- 人 -> 等 。 

仅 对 修改 过 的 模板 文件 进行 重新 编译 。 

模板 中 可 以 使 用 if/elseif/else/endif。 

内 建 缓存 支持 。 

可 自 定义 插件 。 


国 因 办 多国 多 


23.2 ”Smarty 的 安装 配置 
敬 i 视频 讲解 :光盘 \TM\Video\ 第 23 章 \Smarty 的 安装 配置 .exe 


23.2.1 Smarty 下 载 和 安装 


PHP 没有 内 置 Smarty 模板 类 , 需要 单独 下 载 和 配置 , 并且 Smarty 要 求 服务 器 上 的 PHP 版 本 最 低 为 4.0.6。 
用 户 可 以 通过 http://smarty.net/download.php 网 站 下 载 最 新 的 Smarty 压缩 包 。 本 章 使 用 的 版 本 是 Smarty-2.6.26。 

将 压缩 包 解 压 后 ， 有 一 个 libs 目录 ， 其 中 包含 了 Smarty 类 库 的 核心 文件 ， 包 括 smarty.class.php、 
smarty_Compiler.class.php、config File.classphp 和 debug.html 4 个 文件 ， 还 有 internals 和 plug-ins 两 个 目录 。 
将 libs 目录 复制 到 服务 器 根 目录 下 ， 并 为 其 重新 命名 。 一 般 该 目录 的 名 称 为 smarty、class 等 ， 这 里 将 libs 
文件 夹 重新 命名 为 Smarty。 至 此 ，Smarty 模板 安装 完毕 。 


CE 
即 原 libs 目录 。 


局 
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23.2.2 Smarty 配置 


Smarty 模板 引擎 的 配置 步骤 如 下 。 
(1) 确 定 Smarty 目录 的 位 置 . 因 为 Smarty 类 库 是 通用 的 ,每 一 个 项 目 都 可 能 会 使 用 到 它 ,所 以 将 Smarty 
放 到 根 目录 下 。 因 为 本 章 的 所 有 程序 都 放 在 TM/23/ 文 件 夹 下 ， 所 以 将 /23/ 作 为 临时 的 根 目录 ，Smarty 就 放 
到 这 个 目录 下 。 
(2) 新 建 4 个 目录 templates、templates c、configs 和 cache。 因 为 目录 templates 存放 的 是 项 目的 模板 ， 
所 以 有 人 喜欢 将 templates 放 到 Smarty 目录 外 。 这 两 种 方法 没什么 区 别 ， 只 要 设置 的 路 径 正确 即 可 。 
(3) 创建 配置 文件 。 如 果 要 应 用 Smarty 模板 ， 就 一 定 要 包含 Smarty 类 库 和 相关 信息 。 将 配置 信息 写 
到 一 个 文件 中 ， 用 的 时 候 只 要 包含 配置 文件 就 可 以 。 这 里 要 注意 一 点 ， 配 置 文件 中 要 使 用 绝对 路 径 ， 因 为 
服务 器 不 会 知道 文件 在 第 几 层 目录 中 被 调用 。 配 置 文件 完成 后 ， 保 存 到 根 目 录 下 。 不 要 忘记 本 章 中 所 指 的 
根 目录 是 /23/。 配 置 文件 config .php 的 代码 如 下 : 
<?php 
并 “定义 服务 器 的 绝对 路 径 zl 
define('BASE_PATH''E:\AppServiwww\); 
/* 定义 Smarty 目录 的 绝对 路 径 */ 
define('SMARTY_PATH',TM\23\Smarty\\'); 
/加载 Smarty 类 库 文件 */ 
require BASE_PATH.SMARTY_PATH.'Smarty.class.php'; 
/实例 化 一 个 Smarty 对 象 忆 
$smarty = new Smarty; 
人 定义 各 个 目录 的 路 径 。 */ 
$smarty->template_dir = BASE_PATH.SMARTY_PATH.templates/'; 
$smarty->compile_dir = BASE_PATH.SMARTY_PATH.templates_c/' 
$smarty->config_dir = BASE_PATH.SMARTY_PATH.'configs/"; 
$smarty->cache_dir = BASE_PATH.SMARTY_PATH.'cache/"; 
/* ”定义 定 界 符 */ 
li$smarty->left_delimiter = "<{"; 
li$smarty->right_delimiter = "}>"; 


?> 

上 述 配置 文件 的 参数 说 明 如 下 。 

BASE PATH: 指定 服务 器 的 绝对 路 径 。 
SMARTY_PATH: 指定 smarty 目录 的 绝对 路 径 。 
require0: 加 载 Smarty 类 库 文件 Smarty.class.php。 
$smarty: 实例 化 Smarty 对 象 。 

$smarty->template_dir: 定义 模板 目录 存储 位 置 。 
$smarty-> compile_dir: 定义 编译 目录 存储 位 置 。 
$smarty-> config_dir: 定义 配置 文件 存储 位 置 。 

$smarty-> cache dir: 定义 模板 缓存 目录 。 
$smarty->left_delimiter: 定义 Smarty 使 用 的 开始 定 界 符 。 
$smarty->right_delimiter: 定义 Smarty 使 用 的 结束 定 界 符 。 


加 


图 回回 回回 回回 罗 加 


人 注意 有 关 定 界 符 的 使 用 ， 开 发 者 可 以 指定 任意 的 格式 ， 也 可 以 不 指定 。 使 用 Smarty 默认 的 定 界 符 


A 
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到 此 ，Smarty 的 配置 讲解 完毕 。 至 于 将 配置 文件 存储 在 什么 位 置 ， 可 以 根据 实际 情况 而 定 。 
23.2.3 第 一 个 Smarty 程序 


视频 讲解 : 光盘 \ITM\Video\ 第 22 章 \ 第 一 个 Smarty 程序 .exe 
介绍 Smarty 的 下 载 ,安装 和 配置 方法 后 , 接 下 来 就 创建 一 个 Smarty 程序 , 通过 具体 的 操作 来 了 解 Smarty 


的 应 


例 23.1 创建 第 一 个 Smarty 应 用 实例 ， 初 步 了 解 Smarty 的 使 用 过 程 。 (实例 位 置 : 光盘 
\TM\Instance\23\23.1) 

(1) 在 服务 器 根 目 录 下 的 /TM/23/ 文 件 夹 下 ， 新 建 一 个 文件 夹 ， 命 名 为 23.1。 

(2) 复制 Smarty 到 目录 23.1 下 。 在 Smarty 目录 下 新 建 4 个 目录 , 分 别 是 templates、 templates_c、configs 
和 cache。 此 时 ， 例 23.1 的 目录 结构 如 图 23.2 所 示 。 

(3) 新 建 一 个 .html 静态 页 ， 输 入 数据 。 输 入 完毕 后 将 文件 保存 到 刚刚 新 建 的 templates 目录 下 ， 并 命 
名 为 index.html。 实 例 代 码 如 下 : 

<html> 

<head> 

<meta http-equiv="Content-Type" content= "text/html; charset=gb2312" /> 

<title>{ $title }</title> 

</head> 

<body> 
{$content} 


</body> 
</html> 


PR 
< 说 明 代码 中 加 粗 的 部 分 就 是 Smarty 标签 ， 大 括号 “{}” 为 标签 的 定 界 符 ，S$title 和 $content 为 
变量 。 


(4) 回 到 上 级 目录 ， 在 目录 23.1 下 新 建 一 个 .php 文件 ， 使 用 Smarty 变量 和 方法 对 文件 进行 操作 ， 输 

入 完毕 后 保存 为 index.php。 其 代码 如 下 : 

<?php 

上 * ”定义 服务 器 的 绝对 路 径 */ 

define(0BASE_PATH',$_SERVER[DOCUMENT_ROOT]); 

上 定义 Smarty 目录 的 绝对 路 径 */ 

define(SMARTY_PATH'ATMN23\23.TSmartyN); 

请 加 载 Smarty 类 库 文件 */ 

require BASE_PATH.SMARTY_PATH.'Smarty.class.php'; 

* 实例 化 一 个 Smarty 对 象 六 

$smarty = new Smarty; 

上 定义 各 个 目录 的 路 径 。 */ 

$smarty->template_dir = BASE_PATH.SMARTY_PATH.'templates/: 

$smarty->compile_dir = BASE_PATH.SMARTY_PATH.templates_c/' 

$smarty->config_« BASE_PATH.SMARTY_PATH.'configs/’; 

$smarty->cache_dir = BASE_PATH.SMARTY_PATH.'cache/ 

上 使 用 Smarty 赋值 方法 将 指定 数据 发 送 到 模板 中 */ 

$smarty->assign('title',' 第 一 个 Smarty 程序 "); 

$smarty->assign('content','Hello,Welcome to study\'Smarty\!"); 

上 * 显示 模板 */ 

$smarty->display('index.html"); 

?> 


@ 


全 
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这 是 Smarty 运行 最 关键 的 步骤 ， 主 要 进行 了 两 项 设置 和 两 步 操作 。 

加 载 Smarty 类 库 。 也 就 是 加 载 Smarty.class.php 文件 ， 这 里 使 用 的 是 绝对 地 址 。 为 了 稍 后 在 配置 其 
他 路 径 时 不 用 输入 那么 长 的 地 址 字 串 ， 之 前 还 声明 了 两 个 常量 : 服务 器 地 址 常量 和 Smarty 路 径 常 
量 。 两 个 常量 连接 起 来 就 是 Smarty 类 库 所 在 的 目录 。 

保存 新 建 的 4 个 目录 的 绝对 路 径 到 各 自 的 变量 。 在 第 (2) 步 时 ， 曾 创建 了 4 个 目录 。 这 4 个 目录 
各 有 各 的 用 途 ， 如 果 没 有 配置 目录 的 地 址 ， 那 么 服务 器 默认 的 路 径 就 是 当前 执行 文件 所 在 的 路 径 。 
除了 上 面 两 项 必须 设置 的 变量 外 ， 还 可 以 改变 很 多 Smarty 参数 值 ， 如 开启 /关闭 缓存 、 改 变 Smarty 
的 默认 定 界 符 等 ， 这 些 变量 将 在 23.4.2 节 中 介绍 。 

给 模板 赋值 。 设 置 成 功 后 ， 需 要 给 指定 的 模板 赋值 。assign0 就 是 赋值 方法 。 

显示 模板 。 一 切 操作 结束 后 ， 调 用 display0 方 法 来 显示 页 面 。 实 际 上 ， 用 户 真 正 看 到 的 页 面 是 
templates 模板 目录 下 的 index.html 模板 文件 ， 作 为 首页 的 index.php， 只 是 用 来 传递 结果 和 显示 模 
板 ; 而 在 浏览 器 中 运行 的 是 templates_c 文件 夹 下 的 文件 。 

打开 下 浏览 器 ， 运行 index.php 文件 。 运 行 结果 如 图 23.3 所 示 。 


EC 
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图 23.2 Smarty 包 的 目录 结构 23.3 ”第 一 个 Smarty 程序 


23.3 Smarty 模板 设计 


顽 1 视频 讲解 :光盘 \IM\Video\ 第 23 章 \Smarty 模板 设计 .exe 

Smarty 的 特点 是 将 用 户 界面 和 过 程 实现 分 离 ， 让 美工 和 程序 员 各 司 其 职 ， 互 不 干扰 。 这 样 ，Smarty 类 
库 也 自然 被 分 成 两 部 分 来 使 用 ， 即 Smarty 模板 设计 和 Smarty 程序 设计 。 两 部 分 内 容 既 相互 独立 ， 同 时 也 有 
一 部 分 共通 。 本 节 首 先 来 学 习 Smarty 模板 设计 。 


23.3.1 Smarty 模板 文件 

Smarty 模板 文件 是 由 一 个 页 面 中 所 有 的 静态 元 素 ， 加 上 定 界 符 “{…}” 组 成 的 。 模 板 文件 统一 存放 的 
位 置 是 templates 目录 下 (模板 文件 的 存储 位 置 可 以 在 配置 文件 中 指定 ， 可 以 根据 个 人 习惯 而 定 ) 。 模 板 中 
不 允许 出 现 PHP 代码 段 。Smarty 模板 中 的 所 有 注释 、 变 量 、 函 数 等 都 要 包含 在 定 界 符 内 。 
23.3.2 ”注释 


Smarty 中 的 注释 和 PHP 注释 类 似 , 都 不 会 显示 在 源 代码 中 。 注释 包含 在 两 个 星 号 “* ” 中间， 格式 如 下 : 


人 这 是 注释 分 
3 
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23.3.3 ”变量 


Smarty 中 的 变量 来 自 以 下 3 个 部 分 。 
1. PHP 页 面 中 的 变量 
使 用 方法 和 在 PHP 中 是 相同 的 ， 也 需要 使 用 “$” 符 号 ， 略 有 不 同 的 是 对 数组 的 读 取 。Smarty 中 读 取 数 
组 有 两 种 方法 ， 一 种 是 通过 索引 获取 ， 与 PHP 中 相似 ， 可 以 是 一 维 ， 也 可 以 是 多 维 ; 另 一 种 是 通过 键 值 获 
取 数 组 元 素 ， 这 种 方法 的 格式 与 以 前 接触 过 的 不 同 ， 使 用 符号 “.” 作 为 连接 符 ， 数 组 $arr = array{frobject => 
'book','type' => 'comptuer','unit => ' 本 }， 如 果 想 得 到 type 的 值 ， 表 达 式 的 格式 为 $arr.type。 这 个 格式 同样 适用 
于 二 维 数组 。 
例 23.2 使 用 上 述 两 种 方法 读 取 数组 值 。 (实例 位 置 ， 光盘 \TMNInstance\23\23.2) 
代码 如 下 : 
Smarty/templates/2/index.html 文件 
<html> 
<head> 
{ 页 面 的 标题 变量 $title 分 
<title>{ $title }</title> 
</head> 
<body> 
购书 信息 : <p> 
{* ”使 用 索引 取得 数组 的 第 一 个 元 素 值 分 
图 书 类 别 : { $arr[0] }<br /> 
{* 使 用 键 值 取 得 第 二 个 数组 元 素 值 分 
图 书 名 称 : { $arr.name }<br /> 
{* ”使 用 键 值 取得 二 维 数组 的 元 素 值 分 
图 书 单价 : { $arr.unit_price.price XY{ $arr.unit_price.unit } 
</body> 
</html> 
index.php 文件 
<?php 
include_once'../config.php'; 
$smarty->cache_dir = BASE_PATH.SMARTY_PATH.' $arr = array('computerbook','name' => 'PHP 开发 实战 
宝典 ','unit_price' => array('price' => ' 半 65.00','unit => ' 本 ')); 
$smarty->assign('title',' 使 用 Smarty 读 取 数组 '); 
$smarty->assign('arr', $arr); 
$smarty->display('2/index.html'); 


?> 
运行 结果 如 图 23.4 所 示 。 
2. 保留 变量 


相当 于 PHP 中 的 预定 义 变量 。 在 Smarty 模板 中 使 用 保留 变量 时 ， 无 需 使 用 assign() 方 法 传 值 ， 直 接 调 
用 变量 名 即 可 。Smarty 中 常用 的 保留 变量 如 表 23.1 所 示 。 


表 23.1 Smarty 中 常用 的 保留 变量 
get、post、 server、 session、cookie、request | 等 价 于 PHP 中 的 $ GET、$ POST、$_ SEVER、$_ COOKIE、$_ REQUEST 


当前 的 时 间 戳 等 价 于 PHP 中 的 timeO 
用 const 包含 修饰 的 为 常量 


配置 文件 内 容 变量 。 参 见 例 23.4 
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例 23.3 在 模板 文件 中 输出 一 些 保留 变量 的 值 。( 实 例 位 置 ， 光盘 \TMNInstance\23\23.3) 
代码 如 下 : 

templates/3/index.html 文件 

{* 设置 标题 名 称 *} 

<title>{ Stitle }</title> 

<body> 

{ ”使 用 get 变量 获取 url 中 的 变量 值 (ex: http://localhost/tm/23/23.3/index.php?type=computer) 六 
变量 type 的 值 是 : { $smarty.get.type }<br /> 

当前 路 径 为 : { $smarty.server.PHP_SELF}<br /> 

当前 时 间 为 : {$smarty.now} 


</body> 

index.php 文件 

<?php 
include '../config.php'; // 载 入 配置 文件 
$smarty->assign('title''Smarty 保留 变量 '); // 向 模板 中 赋值 
$smarty->display('3/index.html"); /显示 指定 模板 

ea 


运行 结果 如 图 23.5 所 示 。 


99511 
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23.4 使 用 Smarty 读 取 数组 23.5 Smarty 保留 变量 


3. 从 配置 文件 中 读 取 数 据 
Smarty 模板 也 可 以 通过 配置 文件 来 赋值 。 对 于 PHP 开发 人 员 来 说 ， 对 配置 文件 的 使 用 从 安装 服务 器 就 


开始 了 ， 对 文件 的 格式 也 有 了 一 个 初步 的 了 解 。 调 用 配置 文件 中 变量 的 格式 有 以 下 两 种 : 


使 用 “#” 号 。 将 变量 名 置 于 两 个 “#” 号 中 间 ， 即 可 像 普 通 变 量 一 样 调 用 配置 文件 内 容 。 
使 用 保留 变量 中 的 $smarty_config 来 调用 配置 文件 。 

例 23.4 通过 上 面 两 种 方法 来 调用 配置 文件 4.conf 的 内 容 。《〈 实 例 位 置 ， 光盘 \TMNInstance\23\23.4) 
代码 如 下 : 

Smarty/configs/4/4.conf 文件 

title = "调用 配置 文件 " 

bgcolor = "#fOfOfO" 

border = "5" 

type = "计算 机 类 " 

name = "PHP 开发 实战 宝典 " 

templates/4/index.html 文件 

{ config_load file="4/4.conf" } 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 

<title>{#title##</title> 

</head> 

<body bgcolor="{#bgcolor#}"> 

<table border="{#border#"> 
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<tr> 


<td>{$smarty.config.type}</td> 
<td>{$smarty.config.name}</td> 


</tr> 

</table> 
</body> 
</html> 
index.php 文件 
<?php 


include_once '../config.php’; 
$smarty->display('4/index.html'); 


?> 
运行 结果 如 图 23.6 所 示 。 
23.3.4 ”修饰 变量 


23.3.3 节 中 介绍 了 如 何在 Smarty 模板 中 调用 变量 , 但 ”| 到 
有 的 时 候 ， 不 仅 要 取得 变量 值 ， 还 要 对 变量 进行 处 理 。 变 


量 修饰 的 一 般 格式 如 下 : 


计算 机 类 FHP 开发 实战 宝典 


ET 


图 23.6 调用 配置 文件 


{variable_name|modifer_name: parameter1:…} 

variable name 为 变量 名 称 。 

modifer name 为 修饰 变量 的 方法 名 。 变 量 和 方法 之 间 使 用 符号 “|” 分 隔 。 
parameterl 是 参数 值 。 如 果 有 多 个 参数 ， 则 使 用 “:” 分 隔 。 

Smarty 提供 了 修饰 变量 的 方法 。 常 用 的 方法 如 表 23.2 所 示 。 


方 法 名 
capitalize 
count characters:true/false 
cat: "characters" 
date_format: "%Y-%M-%D" 
default: "characters" 


表 23.2 修饰 变量 的 常用 方法 和 说 明 
说 明 
首 字母 大 写 
变量 中 的 字符 串 个 数 。 如 果 后 面 有 参数 tue， 则 空格 也 被 计算 ， 否 则 忽略 空格 
将 cat 中 的 字符 串 添加 到 指定 字符 串 的 后 面 
格式 化 日 期 和 时 间 。 等 同 于 PHP 中 的 strftimeO 函 数 
设置 默认 值 。 当 变量 为 空 时 ， 将 使 用 default 后 面 的 默认 值 
用 于 字符 串 转 码 。value 值 可 以 为 html、htmlall、url、quotes、hex、hexentity 和 


Ce javascript， 默 认为 html 
lower 将 变量 字符 串 改 为 小 写 
nl2br 所 有 的 换行 符 将 被 替换 成 <br/>， 功 能 同 PHP 中 的 nl2br0 函 数 一 样 


regex replace:"parameterl":"value2" 


正则 替换 。 用 value2 蔡 换 所 有 符合 parameterl 标准 的 字 串 


Teplace: "valuel":"value2" 


蔡 换 。 使 用 value2 替换 掉 所 有 valuel 


string format: "Value" 


使 用 value 来 格式 化 字符 串 。 如 果 value 为 %d， 则 字符 串 被 格式 化 为 十 进 制 数 


strip tags 


去 掉 所 有 的 html 标签 


Wr 


将 变量 改 为 大 写 


在 对 变量 进行 修饰 时 ， 不 仅 可 以 单独 使 用 上 面 的 方法 ， 还 可 以 同时 使 用 多 个 。 需 要 注意 的 是 ， 在 每 个 


方法 之 间 使 用 “|” 分 隔 即 可 。 


@ 
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例 23.5 使 用 表 23.2 中 的 几 个 方法 来 修饰 字符 串 。 其 他 方法 的 使 用 , 读者 可 以 自行 练习 。( 实 例 位 置 : 


光盘 \TM\Instance\23\23.5) 


实例 代码 如 下 : 

Templates/5/index.html 文 件 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title>{$title}</title> 

</head> 

<body> 

原文 : {$str} 


<p> 

变量 中 的 字符 数 〈 包 括 空格 ) : {$strlcount_characters:true} 

<br /> 

使 用 变量 修饰 方法 后 : {$strlnl2brlupper} 

</body> 

</html> 

index.php 文件 

<?php 
include_once "../config.php"; 
$str1 = ' 这 是 一 个 实例 。'; 
$str2 = "\n 图 书 -> 计算 机 类 ->php\n 书 名 : 《php 开发 实战 宝典 》"; 
$str3 = "\n 价格 : 站 86/ 本 。"; 
$smarty->assign('title',' 使 用 变量 修饰 方法 '); 
$smarty->assign('str,$str1.$str2.$str3.$str4); 
$smarty->display('S/index.html'); 

?> 


运行 结果 如 图 23.7 所 示 。 


23.3.5 ”流程 控制 


Smarty 模板 中 的 流程 控制 语句 包括 if…elseif…else 条 件 控制 语句 和 foreach、section 循环 控制 语句 。 
1.if.…elseif.…else 语句 
站 条 件 控制 语句 的 使 用 和 PHP 中 的 站 大 同 小 异 。 需要 注意 的 是 , 在 Smarty 模板 中 寺 必 须 以 /站 为 结束 标 


记 。 半 语句 的 格式 如 下 : 


{if 条 件 语句 分 
语句 1 

{elseif 条 件 语句 2} 
语句 2 

{else} 


{iD 
上 述 条 件 语句 中 ， 除 了 可 以 使 用 PHP 中 的 <、>、=、!= 等 常见 运算 符 外 ， 还 可 以 使 用 eq、ne、neq、gt、 


lt, lte、 le、gte、ge、is even、is odd、is not even、is not odd、not、mod、div by、even by、odd by 等 修饰 词 


修饰 。 具 体 含 义 留 给 读者 自己 动手 来 理解 。 


例 23.6 ”使 用 条 件 判 断 语句 选择 不 同 的 返回 信息 。《〈 实 例 位 置 : 光盘 \TMNInstance\23\23.6) 
代码 如 下 : 
templates/6/index.html 文件 


@ 
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<html > 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 

<title>{Stitle}</title> 

<link rel='stylesheet href="../css/style.css" /> 

</head> 

<body> 

<p> 

{if $smarty.get.type == ‘tm'} 

欢迎 光临 ，{$smarty.get.type} 

{else} 

对 不 起 ， 您 不 是 本 站 VIP， 无 权 访 问 此 栏目 。 

Vif 

</body> 

</html> 

index.php 文件 

<?php 
include_once "../config.php"; 
$smarty->assign("title","if 条 件 判断 语句 "); 


$smarty->display("6/index.html"); 
?> 
运行 结果 如 图 23.8 所 示 。 
箱 使 用 安 旺 介 忆 方法 - Windows Interne: Explorer |) 
GO eochowsi To/ sndecphp ox) erez P- 


ET 靖 回 “让 班机/ 工具。 者 有 tH) 
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后 次， 这 是 一 个 去 加。 图 书 -计算 从 phy 书 省 ， 《ow 开 发 详 二 守 下 > 价格 站 867 本 。 

变量 中 的 字符 数 (包括 宝 格 ) ，75 EU 

区 用 称 量 人 作法 尼 ， 世 生 一 个。 IE 

图书 -> 计 半 机 天-PI ss 
人 a 不 是 本 站 VIP， 无 权 访问 此 栏目 

3 篇 丰 地 imranet | 俐 六 区: 驮 月 和 ” 成 100% = 上 全 本 地 ntranet | 出 P 展 式 :名 用 向 成 100% ~ 


图 23.7 使 用 变量 修饰 方法 图 23.8 这 条 件 判 断 语 句 
2. foreach 循环 控制 
Smarty 模板 中 的 foreach 语句 可 以 循环 输出 数组 , 与 另 一 个 循环 控制 语句 section 相 比 , 在 使 用 格式 上 要 
简单 得 多 ， 一 般 用 于 简单 数组 的 处 理 。foreach 语句 的 格式 如 下 : 


{foreach name=foreach_name key=key item=item from=arr_name} 


{/foreach} 

name 为 该 循环 的 名 称 。 

key 为 当前 元 素 的 键 值 。 

回 item 为 当前 元 素 的 变量 名 。 

from 为 该 循环 的 数组 。 

其 中 ，item 和 from 是 必 选 参数 ， 不 可 省 略 。 

例 23.7 使 用 foreach 语句 ,循环 输出 数组 infobook 的 全 部 内 容 .( 实 例 位 置 :光盘 \TM\Instance\23\23.7) 
代码 如 下 : 

templates/7/index.html 文件 

<html> 


局 


第 23 章 ”Smarty 模板 技术 


<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title>{Stitle}</title> 
</head> 
<body> 
使 用 foreach 语句 循环 输出 数组 。<p> 
{foreach key=key item=item from=$infobook} 
{$key} => {$item}<br /> 
{/foreach} 
</body> 
</html> 
index.php 文件 
<?php 
include_once '../config.php'; 
$infobook = array('object'=>'book','type'=>'computer','name'=>'PHP 开发 实战 宝典 ,publishing'=>' 清 华 大 学 
出 版 社 ); 
$smarty->assign('title',' 使 用 foreach 循环 输出 数组 内 容 '); 
$smarty->assign('infobook', $infobook); 
$smarty->display('7/index.html"); 


汉人 
运行 结果 如 图 23.9 所 示 。 
3. section 循环 


Smarty 模板 中 的 另 一 个 循环 语句 是 section， 该 语句 可 用 于 比较 复杂 的 数组 。section 语句 的 语法 如 下 : 
{section name="sec_name" loop=$arr_name start=num step=num} 
name 为 该 循环 的 名 称 。 
loop 为 循环 的 数组 。 
start 表示 循环 的 初始 位 置 ， 如 start=2， 则 说 明 循 环 是 从 loop 数组 的 第 二 个 元 素 开 始 的 。 
step 表示 步 长 ， 如 step=2， 则 循环 一 次 后 数组 的 指针 将 向 下 移动 两 位 ， 依 此 类 推 。 
例 23.8 使 用 section 语句 循环 输出 一 个 二 维 数组 。〈 实 例 位 置 ， 光盘 \TM\Instance\23\23.8) 
代码 如 下 : 
templates/8/index.html 文件 
<html> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" /> 
<title>{$title}</title> 
<link rel="stylesheet" href="../css/style.css" /> 
</head> 
<body> 
<table width="100" border="0" align="left" cellpadding="0" cellspacing="0"> 
{section name=sec1 loop=$obj} 
<tr> 
<td colspan="2">{$obj[sec1].bigclass}</td> 
</tr> 
{section name=sec2 loop=$obj[sec1].smallclass} 
<tr> 
<td width="25">&nbsp;</td> 
<td width="75">{$obj[sec1].smallclass[sec2].s_type}</td> 
</tr> 
{/section} 
{/section} 


491 


PHP+MySQL 开发 实战 


</table> 
</body> 
</html> 
index.php 文件 
<?php 
require "../config.php"; 
$obj = array(array("id" => 1, "bigclass" => "计算 机 图 书 ","smallclass" => array(array("s_id" => 1, "s_type" => 
"PHP"))),array("id" => 2, "bigclass" => "历史 传记 ","smallclass" => array(array("s_id" => 2, "s_type" => "中 国 历史 
"), array("s_id" => 3, "s_type" => "世界 历史 "))),array("id" => 3, "bigclass" => "电子 小 说 ","smallclass" => array 
(array("s_id" => 4, "s_type" => "玄幻 小 说 "),array("s_id" => 5, "s_type" => "言情 小 说 ")))); 
$smarty->assign('title','section 循环 控制 "); 
$smarty->assign("obj", $obj); 
$smarty->display("8/index.html"); 
?> 
运行 结果 如 图 23.10 所 示 。 
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23.9 使 用 foreach 语句 循环 输出 数组 内 容 23.10 ”使 用 section 循环 控制 输出 数组 


ET ”00% ~ 


23.4 ”Smarty 程序 设计 


茹 4 视频 讲解 : 光盘 \TM\Video\ 第 23 章 \Smarty 程序 设计 .exe 

通过 前 面 的 学 习 已 经 知道 ， 在 Smarty 模板 中 ， 是 不 推荐 有 PHP 代码 段 的 ， 所 有 的 PHP 程序 都 要 另 写 
成 文件 。Smarty 程序 的 功能 主要 分 两 种 : 一 种 功能 是 和 Smarty 模板 之 间 的 交互 ， 如 assign()、display0 方 法 ; 
另 一 种 功能 就 是 配置 Smarty 函数 ， 如 变量 template_dir、S$config_dir 等 。 本 节 将 学 习 Smarty 程序 设计 的 其 
他 一 些 方法 和 配置 参数 。 


23.4.1 Smarty 中 的 常用 方法 


1. assign 

assign 用 于 在 模板 被 执行 时 为 模板 变量 赋值 。 语 法 如 下 : 

{assign var=" " value=" "} 

参数 var 是 被 赋值 的 变量 名 ， 参 数 value 是 赋 给 变量 的 值 。 

2. display 

display 方法 用 于 显示 模板 ， 需 要 指定 一 个 合法 的 模板 资源 的 类 型 和 路 径 。 还 可 以 通过 第 二 个 可 选 参数 
指定 一 个 缓存 号 ， 相 关 的 信息 可 以 查看 缓存 。 

void display (string template [, string cache_id [, string compile_id]]) 


@ 
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template 指定 一 个 合法 的 模板 资源 的 类 型 和 路 径 。 

cache id 为 可 选 参数 ， 指 定 一 个 缓存 号 。 

compile_id 为 可 选 参数 ， 指 定编 译 号 。 编译 号 可 以 将 一 个 模板 编译 成 不 同 版 本 使 用 ， 如 针对 不 同 的 
语言 编译 模板 。 编 译 号 的 另外 一 个 作用 是 ， 如 果 存 在 多 个 Stemplate_dir 模板 目录 ， 但 只 有 一 个 
Scompile_dir 编译 后 存档 目录 ， 就 可 以 为 每 一 个 Stemplate_dir 模板 目录 指定 一 个 编译 号 ， 以 避免 相 
同 的 模板 文件 在 编译 后 互相 覆盖 。 相 对 于 在 每 一 次 调用 display0 时 都 指定 编译 号 ,也 可 以 通过 设置 
S$compile id 编译 号 属性 来 一 次 性 设 定 。 


Smarty 中 除了 使 用 assign 和 模板 交互 外 ， 还 有 一 些 比较 常用 的 方法 。 方 法 名 称 和 功能 说 明 如 表 23.3 


所 示 。 


表 23.3 Smarty 程序 设计 常用 方法 和 说 明 
方法 格式 说 明 


Void append (string varname, mixed var [. boolean merge]) | 该 方法 向 数组 中 追加 元 素 
Void clear all assign (0) 清除 所 有 模板 中 的 赋值 
Void clear assign (string var) 清除 一 个 指定 的 赋值 


void config load (string file [, string section]) 


加 载 配置 文件 ， 如 果 有 参数 section， 说 明 只 加 载 配置 文件 
中 相对 应 的 一 段 数据 


string fetch (string template) 返回 模板 的 输出 内 容 ， 但 不 直接 显示 出 来 


获取 指定 配置 变量 的 值 ， 如 果 没 有 参数 ， 则 返回 一 个 所 有 


array get_config vars ([string varname]) 配置 变量 的 数组 


array get_template_vars ([string varname]) 


获取 指定 模板 变量 的 值 ， 如 果 没 有 参数 ， 则 返回 一 个 所 有 
模板 变量 的 数组 


bool template_exists (string template ) 检测 指定 的 模板 是 否 存 在 


23.4.2 ”Smarty 的 配置 变量 


Smarty 中 只 有 一 个 常量 SMARTY _DIR， 用 来 保存 Smarty 类 库 的 完整 路 径 ， 其 他 的 所 有 配置 信息 都 保 
存 到 相应 的 变量 中 。 这 里 将 介绍 包括 前 面 章节 中 接触 过 的 template_dir 等 变量 的 作用 及 设置 。 


回 


S$template_dir: 模板 目录 。 模 板 目录 用 来 存放 Smarty 模板 ， 在 前 面 的 实例 中 ， 所 有 的 .html 文件 都 
是 Smarty 模板 。 模 板 的 后 级 没有 要 求 ， 一 般 都 定义 为 .pl、.html 等 。 
$compile_dir: 编译 目录 。 顾 名 思 义 ， 就 是 编译 后 的 模板 和 PHP 程序 所 生成 的 文件 ， 默 认 路 径 为 当 
前 执行 文件 所 在 的 目录 下 的 templates c 目录。 进入 到 编译 目录 ， 可 以 发 现 许多 
“%%…%9%index.html.php ”格式 的 文件 。 随 便 打开 一 个 这 样 的 文件 可 以 发 现 , 实际 上 Smarty 将 模 
板 和 PHP 程序 又 重新 组 合成 一 个 混 编 页 面 。 
$cache_dir: 缓存 目录 。 用 来 存放 缓存 文件 。 同 样 ， 在 cache 目录 下 可 以 看 到 生成 的 .html 文件 。 如 
果 caching 变量 开启 ， 那 么 Smarty 将 直接 从 这 里 读 取 文件 。 
S$config dir: 配置 目录 。 该 目录 用 来 存放 配置 文件 。 例 23.4 中 所 用 到 的 配置 文件 ， 就 保存 到 这 里 。 
$debugging: 调试 变量 。 该 变量 可 以 打开 调试 控制 台 。 只 要 在 配置 文件 〈configphp ) 中 将 
$smarty->debugging 设 为 true 即 可 使 用 。 
$caching: 缓存 变量 。 该 变量 可 以 开启 缓存 。 只 要 当前 模板 文件 和 配置 文件 未 被 改动 ，Smarty 就 直 
接 从 缓存 目录 中 读 取 缓 存 文件 而 不 重新 编译 模板 。 

国 
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23.5， 本 战 
村 | 视频 讲解 : 光盘 \TM\Video\ 第 23 章 \ 实 战 .exe 


23.5.1 Smarty 模板 中 日 期 、 时 间 的 格式 化 输出 


例 23.9 在 PHP 脚本 中 日 期 、 时 间 的 格式 化 输出 最 常用 的 就 是 date0 函 数 ， 那 么 在 Smarty 模板 中 该 如 
何 完 成 日 期 .时 间 的 输出 呢 ? 这 就 是 本 例 中 要 讲解 的 内 容 , 通过 Smarty 模板 中 的 date_format 函数 完成 日 期 、 
时 间 的 格式 化 输出 。《〈 实 例 位 置 : 光盘 \TMNInstance\23\23.9) 
(1) 创建 system 文件 夹 ， 封 装 Smarty 模板 的 配置 方法 ， 其 具体 内 容 可 以 参考 基础 训练 1， 这 里 不 再 歼 
述 了 。 
(2) 创建 index.php 文件 ， 包 含 Smarty 配置 文件 ， 指 定 Smarty 的 模板 页 index.html。 
(3) 创建 Smarty 模板 页 index.html， 在 index.html 模板 文件 中 插入 date_format 函数 ， 输 出 系统 的 当前 
时 间 。 其 关键 代码 如 下 : 
当前 时 间 : {$smarty.nowldate_format:" %A %B - %e - %Y"} 
<!--{($smarty.nowldate_format} 
{$smarty.nowldate_format:"%A, %B %e, %Y":$times} 
{$smarty.nowldate_format:"%H:%M:%S"} 
{$yesterdayldate_format} 
{$yesterdayldate_format:"%A, %B %e, %Y"} 
> 


运行 结果 如 图 23.11 所 示 。 
23.5.2 ”Smarty 模板 中 的 页 面 设计 


例 23.10 在 Smarty 模板 页 中 同样 可 以 将 网 页 的 头 、 尾 和 主 文件 进行 分 别 存储 ， 从 而 达到 页 面 重用 的 
效果 。 这 就 是 本 例 要 讲解 的 内 容 , 在 Smarty 模板 页 中 设计 网 页 页 面 . (实例 位 置 : 光盘 \TMNInstance\23\23.10) 
代码 如 下 : 
(1) 创建 system 文件 夹 ， 封 装 Smarty 模板 的 配置 方法 ， 这 里 不 再 袭 述 。 
(2) 创建 index.php 文件 ， 应 用 switch 语句 根据 超 链 接 传递 的 变量 值 ， 包 含 不 同 的 PHP 脚本 文件 ， 同 
时 将 对 应 的 模板 页 名 称 赋 给 指定 的 模板 变量 。 其 代码 如 下 : 


<?php 
require("system/system.inc.php"); // 包 含 Smarty 模板 的 配置 类 
if(isset($_GET['caption")X{ // 判 断 超 链接 的 变量 值 
S$caption=$_GET['caption']; 
}else{ 
$caption="™"; 
了 
switch ($caption}{ /根据 switch 语句 实现 不 同 页 面 之 间 的 调整 
case "reg"; 
include "reg.php"; // 包 含 PHP 文 件 
$smarty->assign(admin_phtml',reg.html); /将 对 应 的 静态 页 文件 赋 给 指定 的 模板 变量 
break:; 
case "log"; 
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include "log.php"; 
$smarty->assign(admin_phtml',log.html); 
break; 
case "from"; 
include "for.php"; 
$smarty->assign(admin_phtml',forhtml); 
break; 
case "mes"; 
include "mes.php"; 
$smarty->assign(‘admin_phtml','mes.html"); 
break; 
default: 
include "for.php"; 
$smarty->assign(‘admin_phtml",'for.html'); 
break; 


} 

$smarty->assign("title","Smarty 模板 中 的 页 面 设 计 --".$caption) ; /创建 模板 变量 

$smarty->display("index.html"); /指定 模板 页 

?> 

(3) 创建 index.html 模板 页 ， 应 用 include 函数 载 入 页 面 的 头 文件 (top.html) 、 尾 文件 (bottom.html) 
和 主 文件 。 其 关键 代码 如 下 : 


{$title} 
{include file=top.html} 
{include file=$admin_phtml} 
{include file=right.html} 
{include file=bottom.html} 
(4) 创建 include 函数 加 载 的 模板 文件 和 动态 PHP 文件 。 
运行 结果 如 图 23.12 所 示 。 
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23.11 Smarty 模板 中 日 期 的 格式 化 输出 图 23.12 Smarty 模板 中 的 页 面 设计 


23.5.3 ”网 站 公告 


例 23.11 应 用 Smarty 模板 技术 ， 实 现 输出 网 站 公告 信息 的 功能 。〔( 实 例 位 置 光盘 
\TM\Instance\23\23.11) 
(1) 创建 config.php 文件 , 配置 Smarty 模板 。 指定 服务 器 的 根 目录 、Smarty 模板 在 实例 中 的 存储 位 置 ， 
以 及 各 个 文件 夹 的 存储 位 置 。 其 代码 如 下 : 
@ 
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<?php 

/* ”定义 服务 器 的 绝对 路 径 */ 
define(BASE_PATH',$_SERVER[DOCUMENT_ROOT]): 

上 定义 Smarty 目录 的 绝对 路 径 */ 
define('SMARTY_PATH',\tm\23\23.11\Smarty\\'); 

人 加 载 Smarty 类 库 文件 */ 

require BASE_PATH.SMARTY_PATH.'Smarty.class.php'; 

人 实例 化 一 个 Smarty 对 象 */ 

$smarty = new Smarty; 

/* 定义 各 个 目录 的 路 径 。 */ 

$smarty->template_dir = BASE_PATH.SMARTY_PATH.templates/'; 
$smarty->compile_dir = BASE_PATH.SMARTY_PATH .templates_c/'; 
$smarty->config_dir = BASE_PATH.SMARTY_PATH.'configs/"; 
$smarty->cache_dir = BASE_PATH.SMARTY_PATH.'cache/"; 


?> 
(2) 创建 index.php 文件 ， 从 数据 库 中 读 取出 公告 信息 ,将 公告 信息 存储 到 指定 的 模板 变量 中 ， 并 指定 
<?php 
include_once "conn/conn.php"; /| 连接 数据 库 
include_once "config.php"; // 调 用 配置 文件 
$sql = "select id,title from tb_public order by id"; /| 编写 查询 语句 
Snum = 4; 
S$rst = $conn->SelectLimit($sql,$num); // 执 行 查询 操作 
$arr = $rst->GetAssoc(); // 获 取 结 果 集 
$smarty->assign('arr,$arr); // 将 返回 的 结果 集 存储 到 指定 的 Smarty 模板 变量 中 
$smarty->display('index.html"); /| 执行 模板 文件 
?> 


(3) 在 smarty\templates 文件 夹 下 创建 ndex.html 模板 文件 ， 输 出 模板 变量 中 的 公告 标题 信息 。 其 关键 
代码 如 下 : 
<td width="193" align="|left" valign="top" class="exam"> 
{foreach key=key item=item from=$arr} 
<a href="#" class="Ik" onclick="return showme({$key},'showpub.php');" ><img src="images/man.JPG" 
width="14" height="11" border="0" />{$item}</a><br /> 
{/foreach} 
</td> 
(4) 创建 linksjjs 文件 ， 定 义 showme0 〇 函数， 通过 JavaScript 脚本 调用 showpub.php 文件 ， 查 看 公告 的 
详细 内 容 。showmeO 函 数 的 语法 如 下 : 
/uavaScript Document 
function showme(key,wurlX{ 
var purl = wurl + "?id="+key; 
open(purl,'_blank','width=450 height=200' ,false); 
return false; 


该 函数 的 参数 是 指定 公告 信息 的 ID 和 调用 的 文件 。 
(5) 创建 showpub.php 文件 ， 根 据 传递 的 ID 值 ， 从 数据 库 中 查询 出 数据 的 详细 信息 ， 并 将 数据 添加 到 
指定 的 模板 变量 中 ， 最 后 指定 模板 页 。 

<?php 
include_once 'conn/conn.php'; 
include_once 'config.php'; 
$id = $_GETT'id]; 
$sql = "select * from tb_public where id = "$_GETTid]; 
Srst = $conn->execute($sql); 
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S$arr = $rst->getAssoc(); 
$smarty->assign('title',' 查 看 公告 '); 
$smarty->assign(arr,$arr[$id]); 
$smarty->display(showpub.html); 
?> 
(6) 创建 模板 页 showpub.html， 读 取 模板 变量 中 的 数据 ， 输 出 公告 的 详细 内 容 。 其 代码 如 下 : 
<table width="400" align="center" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td width="70%" height="25" align="center" valign="middle" class="left"> 标 题 : {$arr.title}</td> 
<td width="30%" height="25" align="center" valign="middle" class="right">&nbsp;{$arr.addtime}</td> 
</tr> 
<tr> 
<td height="100" colspan="2" align="left" valign="top" class="all" style=" text-indent: 10px;"><br>&nbsp; 
{$arr.content}</td> 
</tr> 


</table> 
运行 本 实例 ， 将 直接 输出 网 站 公告 的 标题 信息 ， 单 击 某 个 具体 的 标题 ， 将 弹出 一 个 新 的 窗口 ， 输 出 具 

体 公告 的 内 容 ， 如 图 23.13 所 示 。 
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图 23.13 输出 公告 信息 
23.5.4 ”Smarty 模板 中 应 用 正则 表达 式 


例 23.12 通过 Smarty 模板 中 正则 表达 式 的 应 用 在 Smarty 模板 页 完成 字符 串 的 蔡 换 操 作 ， 将 输出 的 会 
员 名 称 “mr” 蔡 换 为 “明日 科技 ”。 (实例 位 置 : 光盘 \TMNInstance\23\23.12) 

regex_replace 变量 Smarty 模板 中 的 正则 表达 式 ， 依 据 正则 表达 式 对 指定 的 字符 串 进 行 匹 配 。 其 包括 两 
个 参数 : 第 一 个 是 指定 的 正则 表达 式 ， 第 二 个 是 指定 的 蔡 换 文本 。 其 应 用 示例 如 下 : 

{$namelregex_replace:"/[mr]/”":"<font color=#FF0000'> 了 明日 科技 </font>"} 

在 这 个 示例 中 ， 将 模板 变量 Sname 中 的 “mr” 蔡 换 为 “<font color=#FF0000> 明 日 科技 </font>”。 

(1) 创 建 system 文件 夹 ,定义 Smarty 文件 夹 ,存储 编译 目录 缓存 目录 ; 创建 类 文件 system.class.inc.php， 
定义 数据 库 连接 、 操 作 类 。 创 建 system.smarty.inc.php 文件 ， 封 装 Smarty 的 配置 类 ， 定 义 类 的 实例 化 文件 
system.inc.php， 完 成 各 个 类 的 实例 化 操作 ， 并 返回 操作 对 象 。 

(2) 创建 index.php 文件 。 首 先 ， 通 过 header0 函 数 设置 页 面 的 编码 格式 ， 初 始 化 session 变量 。 然 后 ， 
根据 session 变量 判断 当前 用 户 是 否 具 有 访问 权限 。 接 着 ， 如 果 具 有 访问 权限 ， 则 应 用 swicth 语句 ， 根 据 超 
级 链接 传递 的 参数 值 ， 完 成 在 不 同 页 面 之 间 的 跳 转 操作 ， 即 通过 include 语句 包含 不 同 的 动态 PHP 文件 ， 同 
时 将 动态 PHP 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 。 最 后 ， 指 定 模板 页 main html。 

(3) 创建 模板 页 main html。 首 先 ， 输 出 PHP 动态 页 中 定义 的 模板 变量 值 ， 包 括 页 面 的 标题 ($title》、 


@ 
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当前 时 间 〈$dates) 和 当前 页 输出 的 模块 类 别 〈$type、$caption) 。 然 后 ， 通 过 include 函数 加 载 模板 变量 
$admin_phtml 传递 的 模板 页 。 最 后 ， 为 后 台 管 理 系统 中 每 个 功能 模块 创建 热点 链接 ， 并 通过 超 链接 的 参数 
传递 数据 ， 同 时 应 用 Smarty 模板 中 的 escape 方法 对 传递 的 参数 值 进 行 编码 。 
(4) 编辑 vip_look.php 文件 ， 分 页 读 取 数 据 库 中 存储 的 会 员 数据 ， 并 且 将 分 页 读 取 返 回 的 变量 赋 给 模 

板 变 量 ， 最 终 指定 模板 页 。 其 关键 代码 如 下 : 

<?php 

require_once("system/system.inc.php"); // 调 用 指定 的 文件 

// 执 行 查询 语句 ， 从 数据 库 中 读 取 商 品 信息 

$info=$admindb->ExecSQL("select count(*) as total1 from tb_name ",$conn); 


$total1=$info[ol[total11]; /| 统计 数据 库 中 数据 总 数 
if(empty($_GET[pages'])==true || is_numeric($_GET['pages"])==falseX ” /| 判断 变量 pages 是 否 为 空 
$page1=1; // 如 果 变 量 为 空 ， 则 赋值 为 1 
jelsef // 如 果 不 为 空 ， 则 获取 变量 的 值 
$page1=intval($_GET[pages']); 
$pagesize1=1; /定义 每 页 显示 3 条 记录 
访 $total1<$pagesize1){ /| 判断 如 果 数 据 库 中 数据 小 于 每 页 显示 的 记录 数 
S$pagecount1=1; // 则 定义 pagecount 变量 的 值 为 1 
jels 


ef 
if$total1%$pagesize1==0){ 
$pagecount1=intval($total1/$pagesize1); // 用 总 的 记录 数 除 以 每 页 显示 的 记录 数 ， 获 取 共 有 几 页 
jelse{ 
$pagecount1=intval($total1/$pagesize1)+1; 
} 


b 
// 将 要 输出 的 数据 赋 给 assign 模板 变量 
$smarty->assign("total1",$total1); 
$smarty->assign("pagesize1",$pagesize1); 
$smarty->assign("page1",$page1); 
$smarty->assign("pagecount1",$pagecount1); 
$array=$admindb->ExecSQL("select * from tb_name order by id desc limit ".($page1-1)*$pagesize1. 
",$pagesize1" ,$conn ); 
if(!$array}{ 


$smarty->assign("iscommo","F"); // 判 断 如 果 执 行 失败 ， 则 输出 模板 变量 iscommo 的 值 为 F 
}else{ 

$smarty->assign("iscommo","T"); // 判 断 如果 执 行 成 功 ， 则 输出 模板 变量 iscommo 的 值 为 T 

$smarty->assign("arr", $array); /定义 模板 变量 arraybbstell， 输 出 数据 库 中 数据 


} 
$smarty->assign(title," 会 员 浏览 "); 
?> 
(5) 编辑 vip_look.html 模板 页 ， 通 过 section 语句 完成 会 员 信息 的 分 页 输出 ， 并 且 应 用 regex_replace 
变量 将 会 员 名称 中 的 “mr” 蔡 换 为 “明日 科技 ”。 其 关键 代码 如 下 : 
{section name=id loop=$arr} 
<tr> 
<td align="center" bgcolor="#FFFFFF">{$arr[id].idj</td> 
<td align="left" bgcolor="#FFFFFF">{$arrlid]j.namelregex_replace:"/[mr]/":"<font color=#FF0000'> 阴 日 科技 
</font>"}</td> 
<td align="left' bgcolor="#FFFFFF">{$arrlid].email}</td> 
<td align="|left" bgcolor="#FFFFFF">{$arrlid].dates}</td> 
</tr> 
{/section} 
运行 结果 如 图 23.14 所 示 。 
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图 23.14 ”Smarty 模板 中 正则 表达 式 的 运用 
23.5.5 j 半 语句 判断 当前 用 户 权限 


例 23.13 在 动态 PHP 文件 中 ,可 以 通过 session 变量 的 值 判断 当前 用 户 是 否 具 有 访问 权限 , 而 在 Smarty 
模板 页 中 不 可 以 使 用 session 变量 ， 那 么 该 如 何 判断 用 户 是 否 具有 访问 权限 呢 ? 这 就 是 在 本 例 中 将 要 讲解 的 
内 容 , 应 用 站 语句 在 Smarty 模板 页 中 判断 用 户 是 否 具有 访问 权限 。( 实 例 位 置 : 光盘 \TM\Instance\23\23.13) 

(1) 创建 system 文件 夹 ， 封 装 Smarty 模板 的 配置 方法 ， 创 建 存储 编译 文件 、 缓 存 文件 和 配置 文件 的 
文件 夹 。 其 具体 内 容 可 以 参考 基础 训练 1， 这 里 不 再 袭 述 了 。 

(2) 创建 index.php 文件 ， 包 含 Smarty 配置 文件 ， 指 定 Smarty 的 模板 页 index.html。 

(3) 创建 Smarty 模板 页 index.html， 设 计 用 户 登录 页 面 ， 添 加 用 户 登 录 的 表单 元 素 ， 并 且 将 用 户 的 登 
录 信 息 提 交 到 index_ok.php 文件 。 

(4) 创建 index_ok.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 判 断 提 交 的 用 户 名 和 密码 是 否 正确 ， 如 果 正 确 
则 通过 display0 方 法 指定 模板 页 main.html， 并 且 为 模板 变量 competence 赋值 为 T; 否则 指定 到 模板 页 
main.html， 为 模板 变量 competence 赋值 为 FE。 其 关键 代码 如 下 : 


<?php 
header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文 件 编码 格式 
include("system/system.inc.php"); // 包 含 配置 文件 


if($_POSTr Submit]!=" && $_POST[user]!=" && $_POST[pass]!="X 
if$_POST[user]=="mr' && $_POST[pass]=="mrsoft")f 


$smarty->assign("competence","T"); /为 模板 变量 赋值 

$smarty->display(main .html); /指定 模板 页 
jelse{ 

$smarty->assign("competence","F"); 

$smarty->display('main.html"); 


jelsef 
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echo "<script>alert(' 用 户 名 和 密码 不 能 为 空 ! '); window.location.href='index.php';</script>"; 
: 
(5) 创建 main html 模板 页 。 首 先 应 用 站 语 句 对 模板 变量 的 值 进行 判断 ， 如 果 值 为 了 则 输出 本 页 内 容 ; 
如 果 值 为 F 则 给 出 提示 信息 。 其 关键 代码 如 下 : 
{if $competence== 
/省略 了 部 分 内 容 
{ 诊 
{if $competence=="F"} 
<script>alert(' 您 没有 权限 访问 ， 请 重新 登录 ');window.location.href="index.php";</script> 
{i 
运行 结果 如 图 23.15 所 示 。 


23.15 让 语句 判断 用 户 是 否 具备 访问 权限 


23.6 人 小 结 


本 章 主要 介绍 了 Smarty 模板 的 安装 、 配 置 及 使 用 ， 并 且 对 Smarty 的 模板 设计 和 程序 设计 进行 了 讲解， 
它们 是 两 个 相辅相成 的 内 容 ， 是 应 用 Smarty 模板 的 基础 。 在 实战 中 讲解 了 Smarty 的 实际 应 用 。 和 希望 通过 本 
章 的 学 习 ， 读 者 能 够 掌握 Smarty 模板 技术 ， 并 且 能 够 将 其 灵活 地 运用 到 实际 的 网 站 开发 中 。 


23.7 学 习 成 果 检 验 


1. 自 定 配置 一 个 Smarty 环境 , 并 将 Smarty 定 界 符 换 成 其 他 符号 (实例 位 置 :光盘 \TM\Instance\23\23.14) 
2. 用 register_object 方法 注册 模板 函数 。〔 实 例 位 置 ， 光 盘 \TM\Instance\23\23.15) 


fs 


ThinkPHP 框架 
( 名 视频 讲解 : 1S3 分 钟 ) 


ThinkPHP 是 一 个 免费 开源 、 快速 .简单 的 面向 对 象 的 轻 量 级 PHP 
开发 框架 ， 遵 循 Apache2 开源 协议 发 布 ， 是 为 了 敏捷 Web 应 用 开发 
和 简化 企业 级 应 用 开发 而 产生 的 。 

ThinkPHP 借鉴 国外 很 多 优秀 的 枉 架 和 模式 ,使 用 面向 对 象 的 开发 
结构 和 MVC 模式， 采用 单一 入 口 模 式 等 ， 融 合 了 Struts 的 Action 思 
想 和 JSP 的 TagLib (标签 库 )、RoR 的 ORM 映射 和 ActiveRecord 模 
式 ， 封 装 了 CURD 和 一 些 常用 操作 ， 在 项 目 配置 、 类 库 导 入 、 模 板 引 
掌 、 查 询 语 言 、 自 动 验 证 、 视 图 模型 、 项 目 编译 、 缓 存 机 制 、5EO 〇 支 
持 、 分 布 式 数据 库 、 多 数据 库 连 接 和 切换 、 认 证 机 制 和 扩展 性 方面 均 
有 独特 的 表现 。 通 过 本 章 的 学 习 ， 读 者 将 对 ThinkPHP 框架 有 深入 的 
认识 ， 并 且 能 够 达到 简单 应 用 的 程度 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 


wm 了 解 ThinkPHP 概述 

WI 了 解 ThinkPHP 的 项 目 目录 结构 
WI 了 解 ThinkPHP 的 控制 器 

mm 了 解 ThinkPHP 的 视图 

MW 掌握 ThinkPHP 项 目 构建 流程 
WI 掌握 ThinkPHP 的 配置 
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24.1 ThinkPHP 简介 


ThinkPHP 可 以 更 方便 和 快捷 地 开发 和 部 署 应 用 。 其 不 仅仅 是 企业 级 应 用 ,任何 PHP 应 用 开发 都 可 以 从 
ThinkPHP 的 简单 和 快速 的 特性 中 受益 。ThinkPHP 本 身 具 有 很 多 的 原创 特性 ， 并且 倡 导 大 道 至 简 , 开发 由 我 
的 开发 理念 ， 用 最 少 的 代码 完成 更 多 的 功能 ， 宗 旨 就 是 让 Web 应 用 开发 更 简单 、 更 快速 。 

ThinkPHP 遵循 Apache2 开源 许可 协议 发 布 ， 意 味 着 可 以 免费 使 用 ThinkPHP， 甚 至 允许 把 基于 ThinkPHP 
开发 的 应 用 开源 或 商业 产品 发 布 /销售 。 


24.1.1 ThinkPHP 框架 的 特点 


ThinkPHP 是 一 个 性 能 卓越 并 且 功 能 丰富 的 轻 量 级 PHP 开发 框架 。 其 宗旨 就 是 让 Web 应 用 开发 更 简单 、 
更 快速 。ThinkPHP 值得 推荐 的 特性 如 下 所 示 。 

类 库 导 入 : ThinkPHP 是 首先 采用 基于 类 库 包 和 命名 空间 的 方式 导入 类 库 ， 让 类 库 导 入 看 起 来 更 加 
简单 清晰 ， 而 且 还 支持 冲突 检测 和 别名 导入 。 为 了 方便 项 目的 跨 平台 移植 ， 系 统 还 可 以 严格 检查 
加 载 文件 的 大 小 写 。 

URL 模式 : 系统 支持 普通 模式 、PATHINFO 模式 、REWRITE 模式 和 兼容 模式 的 URL 方式 ,支持 

不 同 的 服务 器 和 运行 模式 的 部 署 ， 配 合 URL 路 由 功能 ， 可 以 随心 所 欲 地 构建 需要 的 URL 地 址 和 

进行 SEO 优化 工作 。 

编译 机 制 : 独创 的 核心 编译 和 项 目的 动态 编译 机 制 , 有 效 减 少 了 OOP 开发 中 文件 加 载 的 性 能 开销 。 

查询 语言 ， 内 建 丰 富 的 查询 机 制 ， 包 括 组 合 查 询 、 复 合 查询 、 区 间 查 询 、 统 计 查 询 、 定 位 查询 、 
动态 查询 和 原生 查询 ， 让 数据 查询 简洁 高 效 。 

加 ”视图 模型 :轻松 动态 地 创建 数据 库 视图 ， 多 表 查 询 不 再 烦恼 。 

分 组 模块 ， 不 用 担心 大 项 目的 分 工 协 调和 部 署 问 题 ， 分 组 模块 解决 跨 项 目的 难题 。 

模板 引擎 : 系统 内 建 了 一 款 卓越 的 基于 XML 的 编译 型 模板 引擎 支持 两 种 类 型 的 模板 标签 ， 融合 

了 Smarty 和 JSP 标签 库 的 思想 ， 支 持 标签 库 扩展 。 通 过 驱动 还 可 以 支持 Smarty、EaseTemplate、 

TemplateLite、Smart 等 第 三 方 模板 引擎 。 

回 AJAX 支持 : 内置 AJAX 数据 返回 方法 ， 支 持 JSON、XML 和 EVAL 格式 返回 客户 端 ， 并 且 系统 

不 绑 定 任何 AJAX 类 库 ， 可 随意 使 用 自己 熟悉 的 AJAX 类 库 进 行 操作 。 

回 ”缓存 机 制 : 系统 支持 包括 文件 方式 、APC、Db、Memcache、Shmop、Eaccelerator 和 Xcache 在 内 

的 多 种 动态 数据 缓存 类 型 ， 以 及 可 定制 的 静态 缓存 规则 ， 并 提供 了 快捷 方法 进行 存 取 操 作 。 


24.1.2 ”环境 要 求 


[sl 


ThinkPHP 可 以 支持 Windows/UNIX 服务 器 环境 ， 可 运行 于 包括 Apache、IIS 在 内 的 多 种 Web 服务 器 ， 
需要 PHP 5.0 及 以 上 版 本 支持 ， 支 持 MySQL、MsSQL、PgSQL、Sqlite、Oracle 等 数据 库 。 


24.1.3 ”下载 ThinkPHP 框架 


ThinkPHP 是 一 个 免费 开源 、 快捷 、 简 单 的 OOP 轻 量 级 PHP 开发 框架 。 它 遵循 Apache 2 开源 协议 发 布 ， 


局 


第 24 章 ThinkPHP 框架 


是 为 了 敏捷 的 企业 级 开发 而 产生 的 。 获 取 ThinkPHP 的 方式 有 很 多 。 
官方 的 网 站 为 : http://thinkphp.cn。 
SVN 的 下 载 地址 为 : 
完整 版 本 http://thinkphp.googlecode.com/svn/trunk; 
核心 版 本 http://thinkphp.googlecode.com/svn/trunk/ThinkPHP。 
1. 什么 是 MVC 
MVC 是 一 种 经 典 的 程序 设计 理念 , 此 模式 将 应 用 程序 分 为 3 个 部 分 : 模型 层 (Model) 、 视 图 层 (View) 、 
控制 层 (Controller) ，MVC 是 这 3 个 部 分 英文 字母 的 缩写 。 
MN 
rn 
通常 是 程序 中 相对 稳定 的 部 分 ， 重用 率 高 ; 而 与 用 户 交 互 界 面 一 一 视图 层 ， 却 经 常 改变 。 如 果 因 需求 变 
动 而 不 得 不 对 业务 逻辑 代码 修改 ， 或 者 要 在 不 同 的 模块 中 应 用 到 相同 的 功能 而 重复 的 编写 业务 逻辑 代 
码 , 不 仅 降低 整体 程序 开发 的 进度 ,也 会 使 未 来 的 维护 变 得 非常 困难 。 因此 将 业务 有 逻辑 代码 与 外 观 分 离 ， 
将 会 更 方便 地 根据 需求 改进 程序 ， 这 就 是 MVC 设计 模式 。 


在 PHP Web 开发 中 ，MVC 设计 模式 的 各 自 功能 及 相互 关系 如 图 24.1 所 示 。 


Controller 


(控制 层 ) 


图 24.1 MVC 关系 图 


模型 层 (Model) 

模型 层 是 应 用 程序 的 核心 部 分 ， 它 可 以 是 一 个 实体 对 象 或 一 种 业务 逻辑 ， 它 之 所 以 称 为 模型 ， 是 因为 
它 在 应 用 程序 中 有 更 好 的 重用 性 和 扩展 性 。 

回 ”视图 层 (View) 

视图 层 提供 应 用 程序 与 用 户 之 间 的 交互 界面 ， 在 MVC 理论 中 ， 这 一 层 并 不 包含 任何 的 业务 逻辑 ， 仅 提 
供 一 种 与 用 户 交 互 的 视图 。 

回 ”控制 层 (Controller) 

控制 层 用 于 对 程序 中 的 请 求 进行 控制 ， 其 作用 就 像 国家 的 宏观 调控 ， 它 可 以 选择 调用 哪些 视图 或 者 调 
用 哪些 模型 。 

2. 什么 是 CURD 

CURD 是 数据 库 操作 的 缩写 词 。 也 是 几 种 数据 库 操作 技术 的 缩写 ，“C” 代 表 创 建 〈Create) 、“U” 
代表 更 新 Update》 “R” 代 表 读 取 (Read) ，“D” 代 表 删 除 (Delete) 操作 。CURD 定义 了 用 于 处 理 数 
据 的 基本 操作 。 之 所 以 将 CURD 提升 到 一 个 技术 难题 的 高 度 是 因为 完成 一 个 涉及 在 多 个 数据 库 系 统 中 进行 
CURD 操作 的 汇总 相关 的 活动 ， 其 性 能 可 能 会 随 数据 关系 的 变化 而 有 非常 大 的 差异 。 

© 
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CURD 在 具体 的 应 用 中 并 非 一 定 使 用 create、update、read 和 delete 字样 的 方法 ， 但 是 它们 完成 的 功能 
是 一 致 的 。 例 如 ，ThinkPHP 就 是 使 用 add、save、select 和 delete 方法 表示 模型 的 CURD 操作 。 

3. 什么 是 单一 入 口 

单一 入 口 通常 是 指 一 个 项 目 或 者 应 用 具有 一 个 统一 (但 并 不 一 定 是 唯一 ) 的 入 口 文件 ， 也 就 是 说 项 目 
的 所 有 功能 操作 都 是 通过 这 个 入 口 文件 进行 的 ， 并 且 往 往 入 口 文件 是 第 一 步 被 执行 的 。 

单一 入 口 的 好 处 是 项 目 整 体 比 较 规 范 ， 因 为 同一 个 入 口 ， 往 往 其 不 同 操作 之 间 具 有 相同 的 规则 。 另 外 
一 个 方面 就 是 单一 入 口 控制 较为 灵活 ， 因 为 拦截 方便 ， 如 一 些 权 限 控制 、 用 户 登录 方面 的 判断 和 操作 可 以 
统一 处 理 。 


24.2 ThinkPHP 架构 


殴 il 视频 讲解 : 光盘 \TM\Video\ 第 24 章 \ThinkPHP 架构 .exe 
ThinkPHP 遵循 简洁 实用 的 设计 原则 ， 兼 顾 开发 速度 和 执行 速度 的 同时 ， 也 注重 易 用 性 。 本 节 内 容 将 对 
ThinkPHP 框架 的 整体 思想 和 架构 体系 进行 详细 说 明 。 本 章 使 用 的 版 本 是 ThinkPHP2.0。 


24.2.1 ThinkPHP 的 目录 结构 


ThinkPHP 框架 中 目录 分 为 两 部 分 : 系统 目录 和 项 目 目录 。 系 统 目 录 是 下 载 的 ThinkPHP 框架 类 库 本 身 
的 ， 如 表 24.1 所 示 。 


表 24.1 系统 目录 
目录 名 称 主要 作用 
Common 包含 框架 的 一 些 公 共 文件 、 系 统 定义 和 惯例 配置 等 
Lang 目录 语言 文件 来 ， 目 前 ThinkPHP 支持 的 语言 包 有 简体 中 文 、 繁 体 中文 、 英 文 
Lib 系统 的 基 类 库 目录 
Tpl 系统 的 模板 目录 
Mode 框架 模式 扩展 目录 
Vendor 第 三 方 类 库 目 录 


项 目 目录 是 用 户 实际 应 用 的 目录 ， 如 表 24.2 所 示 〈ThinkPHP 采用 自动 创建 文件 夹 的 机 制 ， 当 用 户 布置 
好 ThinkPHP 的 核心 类 库 后 ， 编 写 运行 入 口 文件 ， 则 相关 应 用 到 的 项 目 目录 就 会 自动 生成 ) 。 


表 24.2 项 目 目录 
目录 名 称 主要 作用 
indexphp | 项 目 入 口 文件 
Common | 项 目 公共 目录 ， 放 置 项 目 公共 函数 
Tan 项 目 语言 包 目录 (可 选 ) 
Conf 项 目 配置 目录 ， 放 置 配置 文件 
Lib | 项 目 基 目录 ， 通 常 包括 Action 和 Model 目录 
Tpl | 项 目 模板 目录 
Runtime 项 目 运 行 时 目录 ， 包 括 Cache、Temp、Data 和 Log 
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24.2.2 ”自动 生成 目录 


下 面 通过 一 个 实例 ， 讲 解 在 ThinkPHP 框架 中 如 何 自 动 生成 项 目 目录 。 

例 24.1 创建 名 称 为 TM 的 项 目 ， 自 动 生成 项 目 目录 。〔 实 例 位 置 ， 光盘 \TMNInstance\24\24.1) 
其 操作 步骤 如 下 。 

(1) 在 网 站 根 目录 下 创建 文件 夹 ， 并 命名 为 TM。 

(2) 将 ThinkPHP 核心 类 库存 储 于 TM 目录 下 。 

(3) 编写 入 口 文件 mdex.php， 将 其 存储 于 TM 目录 下 。index.php 文件 代码 如 下 : 


<?php 

define('THINK_PATH', ThinkPHP"); /定义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 
define('APP_NAME', TM'); /定义 项 目 名 称 

define(APP_PATH',' /定义 项 目 路 径 
require(THINK_PATH."/ThinkPHP.php"); 1/ 加 载 框架 入 口 文件 

App::run(); /实例 化 一 个 网 站 应 用 实例 

?> 


在 运行 此 文件 前 ， 查 看 TM 项 目的 文件 夹 架构 ， 如 图 24.2 所 示 。 
在 正 浏览 器 中 运行 此 项 目 ， 将 输出 如 图 24.3 所 示 的 运行 结果 ， 此 为 ThinkPHP 提供 的 测试 内 容 。 此 时 
再 次 查看 例 24.1 的 项 目 文件 夹 ， 如 图 24.4 所 示 ， 在 项 目 根 目 录 下 自动 生成 项 目 目录 。 
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24.2.3 ”项 目 目录 部 署 方案 


在 实际 开发 过 程 中 ， 目 录 结 构 往往 由 于 项 目的 复杂 而 变 得 复杂 。 这 里 向 大 家 推荐 两 套 标准 的 目录 部 署 
方案 。 
方案 一 如 图 24.5 所 示 。 
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方案 二 采用 分 组 模块 ， 如 图 24.6 所 示 。 


EEA Boi 


图 24.5 项 目 部 署 方案 一 图 24.6 项 目 部 署 方案 二 


这 样 部 署 的 好 处 是 系统 目录 和 项 目 目录 可 以 存储 于 非 Web 访问 目录 下 面 ,网 站 目录 下 面具 需 放 置 Public 
公共 目录 和 index.php 入 口 文件 〈 如 果 是 多 个 项 目的 话 ， 每 个 项 目的 入 口 文件 都 需要 放 到 Web 目录 下 面 ) ， 
从 而 提高 网 站 的 安全 性 。 


24.2.4 ”命名 规范 


ThinkPHP 框架 有 其 自身 的 一 定 规范 ， 要 应 用 ThinkPHP 框架 开发 项 目 ， 那 么 就 要 尽量 遵守 它 的 规范 。 
下 面 就 介绍 一 下 ThinkPHP 的 命名 规范 。 


图 图 图 回回 图 网 图 加 
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类 文件 都 是 以 .class.php 为 后 缀 〈 这 里 指 的 是 ThinkPHP 内 部 使 用 的 类 库 文件 ， 不 代表 外 部 加 载 的 
类 库 文件 ) ， 使 用 驼峰 法 命名 ， 并 且 首 字母 大 写 ， 如 DbMysql.class.php。 

函数 、 配 置 文 件 等 其 他 类 库 文件 之 外 的 一 般 是 以 .php 为 后 缀 〈 第 三 方 引入 的 不 做 要 求 ) 。 

确保 文件 的 命名 和 调用 大 小 写 一 致 ， 是 由 于 在 类 UNIX 系统 中 ， 对 大 小 写 是 敏感 的 (而 ThinkPHP 
在 调试 模式 下 面 ， 即 使 在 Windows 平台 也 会 严格 检查 大 小 写 ) 。 

类 名 和 文件 名 一 致 〈 包 括 上 面 说 的 大 小 写 一 致 ) ， 如 UserAction 类 的 文件 命名 是 
UserAction.class.php，InfoModel 类 的 文件 名 是 InfoModel.class.php。 

函数 的 命名 使 用 小 写字 母 和 下 划 线 的 方式 ， 如 get_client_ip。 

Action 控制 器 类 以 Action 为 后 级， 如 UserAction、InfoAction。 

模型 类 以 Model 为 后 级 ， 如 UserModel、InfoModel。 

方法 的 命名 使 用 驼峰 法 ， 并 且 首 字母 小 写 ， 如 getUserName。 

属性 的 命名 使 用 驼峰 法 ， 并 且 首 字母 小 写 ， 如 tableName。 

以 双 下 划 线 (_) 打头 的 函数 或 方法 作为 魔法 方法 ， 如 _call 和 autoload。 

常量 以 大 写字 母 和 下 划 线 命名 ， 如 HAS_ONE 和 MANY TO_MANY。 

配置 参数 以 大 写字 母 和 下 划 线 命名 ， 如 HIML _CACHE _ON。 

语言 变量 以 大 写字 母 和 下 划 线 命名 ， 如 MY_LANG， 以 下 划 线 开头 的 语言 变量 通常 用 于 系统 语言 
变量 ， 如 CLASS _ NOT EXIST 。 

数据 表 和 字段 采用 小 写 加 下 划 线 方式 命名 ， 如 think_user 和 user_name。 


和 在 ThinkPHP 中 有 一 个 函数 命名 很 特例 ， 就 是 单字 母 大 写 函 数 ， 这 类 函数 通常 是 菜 些 操作 的 快 
捷 定 义 ， 或 者 有 特殊 的 作用 。 例如 ，ADSL 方法 等 ， 它 们 有 着 特殊 的 含义 。 另 外 一 点 ，ThinkPHP 默认 使 
用 UTF-8 编码 ， 所 以 请 确保 程序 文件 采用 UTF-8 编码 格式 保存 ， 并 且 去 掉 BOM 信息 头 (去 掉 BOM 头 
信息 有 很 多 方式 ， 不 同 的 编辑 器 都 有 设置 方法 ， 也 可 以 用 工具 进行 统一 检测 和 处 理 ) 。 


@ 
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ThinkPHP 具有 项 目 目录 自动 创建 功能 ， 因 此 构建 项 目 应 用 程序 非常 简单 ， 用 户 只 需 定义 好 项 目的 入 口 
文件 ， 在 第 一 次 访问 入 口 文件 时 ， 系 统 会 自动 根据 在 入 口 文件 中 所 定义 的 目录 路 径 ， 迅 速 创 建 好 项 目的 相 
关 目 录 结 构 。 在 完成 项 目 目录 结构 的 创建 后 ， 再 进行 其 他 工作 , 如 图 24.7 所 示 展 示 了 ThinkPHP 创建 项 目的 


基本 流程 。 


例 24.2 ”根据 上 述 讲解 的 流程 , 创建 一 个 名 称 为 TM 的 项 目 , 读 取 db _database24 数据 库 中 的 数据 。( 实 


例 位置 ， 光盘 \TMNInstance\24\24.2) 
其 操作 步骤 如 下 。 


(1) 创建 db_database24 数据 库 和 think_user 数据 表 。 数 据 表 结构 如 图 24.8 所 示 


创建 数据 库 D 项 目 命名 
数据 表 创建 入 口 文件 


了 


创建 控制 关 | < | 项目 本 有 


Y 


创建 模 音 类 ”上 [> 一 | 创建 模板 文件 


人 ， 


图 24.7 ThinkPHP 项 目 构建 流程 
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图 24.8 ThinkPHP 项 目 构建 流程 


(2) 载 入 ThinkPHP 系统 文件 ， 编 辑 入 口 文 件 index.php， 创 建 名 称 为 TM 的 项 目 。index.php 的 代码 


如 下 : 
<?php 
define('THINK_PATH', '../ThinkPHP/); 
define('APP_NAME', TM'): 
define(APP_PATH',' 
require(THINK_PATH."/ThinkPHP.php"); 
App::run();// 实 例 化 一 个 网 站 应 用 实例 
?> 


/定义 ThinkPHP 框架 路 径 
/定义 项 目 名 称 和 路 径 
/定义 项 目 名 称 和 路 径 
/加载 框 架 入 口 文件 


(3) 自动 生成 的 项 目 目录 中 已 经 创建 了 一 个 空 的 项 目 配置 文件 ， 位 于 项 目的 Conf 目录 下 面 ， 名 称 是 
config.php。 重 新 编辑 此 文件 ， 完 成 数据 库 的 配置 。config.php 文件 的 代码 如 下 : 


<?php 

return array( 
'APP_DEBUG' => true, 
'DB_TYPE'=> 'mysql', 
'DB_HOST'=> localhost', 
'DB_NAME'=>'db_database24', 
'DB_USER'=>'root’, 
'DB_PWD'=>"111", 
'DB_PORT'=>'3306', 
'DB_PREFIX'=>'think_', 


// 开 启 调试 模式 
/数据 库 类 型 
/数据 库 服务 器 地 址 
/数据 库 名称 
/数据 库 用 户 名 

/| 数据 库 密码 

/| 数据 库 端口 

// 数 据 表 前 组 
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攻 
?> 
(4) 在 项 目的 Lib\Action 目录 下 ， 定 位 到 自动 生成 的 IndexAction.class.php 文件 ， 这 是 ThinkPHP 的 控 

制 器 ， 即 Index 模块 。 重 新 编辑 控制 器 的 index 方法 ， 查 询 指定 数据 表 中 的 数据 ， 并 且 完 成 数据 的 循环 输出 。 
其 代码 如 下 : 

<?php 

class IndexAction extends Action{ 

public function index() { 


$db = new Model(user); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
$select = $db->select(); // 查 询 数 据 
S$this->assign('select', $select); // 模 板 变量 赋值 
S$this->display(); // 输 出 模板 
} 
} 
?> 


(5) 在 项 目的 Tpl\default 目录 下 ， 创 建 mdex 目录 ， 存 储 Index 模块 的 模板 文件 index.html， 完 成 数据 

库 中 数据 的 循环 输出 。 其 代码 如 下 : 

<!- 循 环 输出 查询 结果 数据 集 --> 

<volist name='select id='user > 

ID:{$user.id}<br/> 

用 户 名 : {$user.user}j<br/> 

地 址 : {$user.address}<hr> 

</volist> 


(6) 在 正 浏 览 器 中 输入 http://127.0.0.1:82/TM/24/24.2/， 其 运行 结果 如 图 24.9 所 示 。 
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24.9 了 解 ThinkPHP 项 目 构建 流程 


24.3 ”ThinkPHP 的 配置 


句 4 视频 讲解 :光盘 \IM\Video\ 第 24 章 \ThinkPHP 的 配置 .exe 

配置 文件 是 ThinkPHP 框架 程序 得 以 运行 的 基础 条 件 ， 框 架 的 很 多 功能 都 需要 在 配置 文件 中 配置 后 , 才 
可 以 生效 ,包括 URL 路 由 功能 、 页 面 伪 静 态 和 静态 化 等 。ThinkPHP 提供 了 灵活 的 全 局 配置 功能 ， 采 用 最 有 
效率 的 PHP 返回 数组 方式 定义 ， 支 持 惯 例 配 置 、 项 目 配置 、 调 试 配 置 和 模块 配置 ， 并 且 会 自动 生成 配置 组 
存 文 件 ， 无 需 重复 解析 。 

ThinkPHP 在 项 目 配置 上 面 创造 了 自己 独 有 的 分 层 配置 模式 ， 其 配置 层次 如 图 24.10 所 示 。 


@ 
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图 24.10 分 层 配置 模式 的 顺序 


以 上 是 配置 文件 的 加 载 顺序 ， 但 是 因为 后 面 的 配置 会 覆盖 之 前 的 配置 (在 没有 生效 的 前 提 下 ) ， 所 以 
优先 顺序 从 右 到 左 。 系 统 的 配置 参数 是 通过 静态 变量 全 局 存 取 的 ， 存 取 方 式 非常 得 简单 、 高 效 。 


24.3.1 配置 格式 


ThinkPHP 框架 中 所 有 配置 文件 的 定义 格式 均 采用 返回 PHP 数组 的 方式 。 其 格式 如 下 : 
<?php 
return array( 

'APP_DEBUG' => true, 

'URL_MODEL' => 2, 

eee // 更 多 的 配置 参数 


/. 
说 日 
Wu 明 配置 参数 不 区 分 大 小 写 ( 因为 无 论 使 用 大 写 或 小 写 定义 ， 都 会 转换 成 小 写 )， 但 是 习惯 上 保持 
大 写 定义 的 原则 。 另 外 ， 还 可 以 在 配置 文件 中 使 用 二 维 数组 来 配置 更 多 的 信息 。 例 如 : 
<?php 
return array( 

'APP_DEBUG' => true, 

'USER_CONFIG' => array( 
'USER_AUTH' => true, 
"USER_TYPE' => 2, 


)， 


?> 


系统 目前 最 多 支持 二 维 数组 的 配置 级 别 , 每 个 项 目 配置 文件 除了 定义 ThinkPHP 所 需要 的 配置 参数 之 外 ， 
开发 人 员 还 可 以 在 里 面 添加 项 目 需要 的 一 些 配置 参数 ， 用 于 自己 的 应 用 。 项 目 配置 文件 默认 存储 于 项 目的 
Conf 目录 。 例 如 ， 在 例 24.2 中 ， 连 接 数 据 库 的 配置 文件 存储 于 项 目的 24.2\Conf\config.php 文件 中 。 


| 

Ls 技巧 项 目 配置 指 的 是 项 目的 全 局 配置 ， 因 为 一 个 项 目 除了 可 以 定义 项 目 配置 文件 之 外 , 还 可 以 定义 
模块 配置 文件 ， 用 于 针对 某 个 特定 的 模块 进行 特殊 的 配置 。 它 们 的 定义 格式 都 是 一 致 的 ， 区 别 只 是 配置 
文件 命名 的 不 同 。 系 统 会 自动 在 不 同 的 阶段 读 取 配置 文件 。 


24.3.2 ”调试 配置 


如 果 启 用 调试 模式 的 话 ， 那 么 会 导入 框架 默认 的 调试 配置 文件 ， 默 认 的 调试 配置 文件 位 于 
Think\Common\debug.php 文件 中 ， 如 果 没 有 检测 到 项 目的 调试 配置 文件 ， 就 会 直接 使 用 默认 的 调试 配置 参 


@ 
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数 。 项 目 定义 自身 的 调试 配置 文件 ， 则 会 和 默认 的 调试 配置 文件 合并 ， 也 就 是 说 ， 项 目 配置 文件 也 只 需要 
配置 和 默认 调试 配置 不 同 的 参数 或 者 新 增 的 参数 。 
调试 配置 文件 也 位 于 项 目 配置 目录 下 面 ， 文 件 名 是 debug.php。 通 常情 况 下 ， 调 试 配置 文件 里 面 可 以 进 
行 一 些 开 发 模式 所 需要 的 配置 。 例 如 ， 配 置 额外 的 数据 库 连接 用 于 调试 、 开 启 日 志 写 入 便于 查找 错误 信息 、 
开启 页 面 Trace 输出 更 多 的 调试 信息 等 。 系 统 默认 的 调试 配置 文件 中 设置 如 下 内 容 。 
开启 日 志 记 录 。 
关闭 模板 缓存 。 
记录 SQL 日 志 。 
关闭 字段 缓存 。 
开启 运行 时 间 详细 显示 〈 包 括 内 存 、 缓 存 情 况 ) 。 
开启 页 面 Trace 信息 显示 。 
严格 检查 文件 大 小 写 〈 即 使 是 Windows 平台 ) 。 


办 办 办 办 办 多 


24.4 ThinkPHP 的 控制 器 
茹 4 视频 讲解 : 光盘 \TM\Video\ 第 24 章 \ThinkPHP 的 控制 器 .exe 


24.4.1 控制 器 


ThinkPHP 的 控制 器 就 是 模块 类 ， 通 常 位 于 项 目的 Lib\Action 目录 下 面 。 类 名 就 是 模块 名 加 上 Action 后 
组 ， 如 IndexAction 类 表示 Index 模块 。 控 制 器 类 必须 继承 系统 的 Action 基础 类 ， 这 样 才能 确保 使 用 Action 
类 内 置 的 方法 。 
例 24.3 对 自动 生成 的 项 目 目录 中 的 控制 器 进行 修改 ， 使 其 输出 自己 编译 的 内 容 。《〈 实 例 位 置 ， 光盘 \ 
TM\Instance\24\24.3) 
其 操作 步骤 如 下 。 
(1) 创建 名 称 为 TM 的 项 目 ， 将 ThinkPHP 核心 类 库存 储 于 TM 目录 下 。 
(2) 编写 入 口 文件 ndex.php， 将 其 存储 于 TM 目录 下 。index.php 文件 代码 如 下 : 
<?php 


define(THINK_PATH'，../ThinkPHP/); /定义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 
define('APP_NAME', TM'); // 定 义 项 目 名 称 

define(APP_PATH,', "."); // 定 义 错误 提示 
require(THINK_PATH."/ThinkPHP.php"); /加载 框架 入 口 文件 

App::run(); /实例 化 一 个 网 站 应 用 实例 

?> 


(3) 运行 index.php 文件 ， 在 TM 目录 下 自动 生成 项 目 目录 ， 其 运行 效果 如 图 24.11 所 示 。 
(4) 在 默认 生成 的 项 目 目录 中 ， 控 制 器 IndexAction 中 输出 的 是 ThinkPHP 设置 的 内 容 ， 此 时 我 们 对 这 
个 内 容 进 行 修改 ， 输 出 “明日 科技 欢迎 您 ! ”。 修 改 后 24.3\Lib\Action\IndexAction.class.php 文件 的 代码 
如 下 : 
<?php 
class IndexAction extends Action{ 
public function index(){ 
header("Content-Type:text/html; charset=utf-8"); ll 设置 编码 格式 
echo "<div style='font-weight:normal;color:blue;float:left:width:345px;text-align:center;border:1px solid 
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silver;background:#E8EFFF;padding:8px;font-size:14px;font-family:Tahoma'>^_^ <span style='font-weight:bold; 
color:red'> 明日 科技 欢迎 您 ! </span></div>"; /| 输出 内 容 
?> 


在 对 控制 器 的 内 容 进行 修改 后 ， 重 新 运行 项 目 ， 在 正 浏览 器 中 输入 http://127.0.0.1:82/TM/24/24.3/， 将 
输出 如 图 24.12 所 示 的 效果 。 


人 人 ^ Helo 允 好 使 用 ThinkpHP | EE TE SET 


图 24.11 自动 创建 项 目 目录 24.12 输出 控制 器 中 的 内 容 


SS 每 个 模块 的 操作 并 非 一 定 要 定义 操作 方法 ， 如 果 只 是 希望 输出 一 个 模板 , 既 没 有 变量 也 没有 任 
何 的 业务 逻辑 ， 那 么 只 要 按照 规则 定义 好 操作 对 应 的 模板 文件 即 可 ， 而 不 需要 定义 操作 方法 。 例如， 在 
IndexAction 中 如 果 没 有 定义 help 方 法 ,但 是 存在 对 应 的 Index/help.html 模板 文件 ,那么 http://localhost:82/ 
24/24.3/index.php/Index/help/ 依 然 可 以 正常 运作 ， 因 为 系统 找 不 到 IndexAction 类 的 help 方法 ， 会 自动 定 
位 到 Index 模块 的 模板 目录 中 查找 help.html 模板 文件 ， 然 后 直接 输出 。 代 码 中 加 粗 的 部 分 就 是 Smarty 
标签 ， 大 括号 “{}” 为 标签 的 定 界 符 ，Stitle 和 $content 为 变量 。 


24.4.2 ” 跨 模块 调用 


在 开发 过 程 中 经 常会 在 当前 模块 调用 其 他 模块 的 方法 ， 这 个 时 候 就 涉及 到 跨 模块 调用 。 下 面 通过 A 和 
RR 两 个 快捷 方法 完成 跨 模块 调用 。 


S$User = A("User"); /实例 化 UserAction 控制 器 对 象 
S$User->insert(); 1/ 调用 User 模块 的 importUser 操作 方法 
这 里 的 A("User") 是 一 个 快捷 方法 ， 与 下 面 的 代码 等 效 : 
import("@.Action.UserAction"); 


S$User = new UserAction(); 
事实 上 ， 在 这 个 例子 里 面 还 有 比 A 方法 更 简单 的 调用 方法 ， 例 如 : 


R("User","insert"); /远程 调用 UserAction 控制 器 的 insert 操作 方法 

上 面 只 是 在 当前 项 目 中 调用 ， 如 果 需 要 在 多 个 项 目 之 间 调 用 方法 ， 一 样 可 以 完成 。 

$User = A("User","Admin"); /实例 化 Admin 项 目的 UserAction 控制 器 对 象 

$User->insert(); /调用 Admin 项 目 UserAction 控制 器 的 insert 操作 方法 
R("User","insert","Admin"); /远程 调用 Admin 项 目的 UserAction 控制 器 的 insert 操作 方法 


例 24.4 ”应 用 跨 模 块 调用 的 方法 , 在 前 台 控制 器 中 调用 后 台 项 目 中 的 insert 方法 完成 用 户 信息 的 添加 操 
作 。《〈 实 例 位 置 ， 光盘 \TM\Instance\24\24.4) 
其 操作 步骤 如 下 。 
(1) 创 建 TM 项 目 根 目录 ,在 根 目录 下 分 别 创建 前 台 项 目 文件 来 Home、 后 台 项 目 文件 夹 Admin 和 Public 
存储 CSS、 图 片 和 JS 脚本 等 文件 。 
(2) 在 TM 项 目 根 目录 下 , 编辑 index.php 前 台 入 口 文件 和 admin.php 后 台 入 口 文件 。 其 关键 代码 如 下 : 
lIndex.php 前 台 入 口 文件 
<?php 
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define(THINK_PATH'，..ThinkPHP/); 
define('APP_NAME', T™'); 
define('APP_PATH', 'Home’); 
require(THINK_PATH."/ThinkPHP.php"); 
App::run(); 

?> 

/Admin.php 后 台 入 口 文件 

<?php 

define('THINK_PATH", '../ThinkPHP/"); 
define('APP_NAME’, TM"); 
define('APP_PATH', Admin'); 
require(THINK_PATH."/ThinkPHP.php"); 
App::run(); 

2 


/定义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 
/定义 项 目 名 称 

/定义 项 目 路 径 

/加 载 框架 入 口 文件 

// 实 例 化 一 个 网 站 应 用 实例 


// 定 义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 
/定义 项 目 名 称 

/定义 项 目 路 径 

// 加 载 框架 入 口 文件 

// 实 例 化 一 个 网 站 应 用 实例 


(3) 在 正 浏 览 器 中 运行 前 后 台 的 入 口 文件 ， 自 动 生成 项 目 目 录 。 
(4) 定位 到 Admin\Conf 目录 下 ， 编 辑 config.php 文件 ， 完 成 后 台 项 目 中 数据 库 的 配置 。 其 代码 如 下 : 


<?php 

return array( 
'APP_DEBUG' => false, 
'DB_TYPE'=> ,mysql'， 
"DB_HOST=> '\ocalhost 
'DB_NAME'=>'db_database24', 
'DB_USER'=>'root', 
'DB_PWD'=>"111", 
'DB_PORT'=>'3306', 
'DB_PREFIX'=>'think_", 

); 


> 


/关闭 调试 模式 
/数据 库 类 型 
/数据 库 服务 器 地 址 
/数据 库 名 称 

/| 数据 库 用 户 名 

/| 数据 库 密码 

/| 数据 库 端 口 

/| 数据 表 前 缀 


《5) 定 位 到 Admin\Lib\Action 目录 下 , 编写 后 台 项 目的 控制 器 。 首 先 创建 mdex 模块 , 继承 系统 的 Action 
基础 类 ， 定 义 index 方法 读 取 指定 数据 表 中 的 数据 ， 并 且 将 查询 结果 赋 给 模板 变量 ， 最 终 指 定 模板 页 。 


IndexAction.class.php 的 代码 如 下 : 
<?php 


header("Content-Type:text/html; charset=utf-8"); 


class IndexAction extends Action{ 
public function index() { 
$db = new Model(user); 
$select = $db->select(); 
S$this->assign('select', $select); 
Sthis->display(); 


} 


?> 


/设置 页 面 编码 格式 


/实例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
/查询 数据 

/模板 变量 赋值 

// 输 出 模板 


然后 创建 User 模块 ， 同 样 继承 系统 的 Action 基础 类 ， 定 义 insert 方法 ， 实 例 化 模型 类 ， 将 表单 中 提交 
的 数据 添加 到 指定 的 数据 表 中 ， 添 加 成 功 后 重 定向 到 后 台 主 页 。UserAction.class.php 的 代码 如 下 : 


<?php 


header("Content-Type:text/html; charset=utf-8"); 


class UserAction extends Action{ 
public function insert() { 
Sins = new Model(user); 
S$ins->Create(); 
Sresult = $ins->add(); 


// 设 置 页 面 编码 格式 

/定义 类 ， 继 承 基础 类 

/定义 方法 

// 实 例 化 模型 类 ， 传 递 参数 为 没有 前 缀 的 数据 表 名 称 
// 创 建 数据 对 象 

// 写 入 数据 库 


Sthis->redirect('Index/index',", 5,' 页 面 跳 转 中 '); /页 面 重 定向 


@_ 


第 24 章 ThinkPHP 框架 


} 
} 
?> 
(6) 定位 到 Admin\Tpl\default 目录 下 ， 首 先 创建 Index 模块 文件 夹 ， 编 辑 index 操作 的 模板 文件 
index.html， 循 环 输出 模板 变量 传递 的 数据 。index.html 模板 文件 的 关键 代码 如 下 : 
<volist name='select id='user > 
<tr class="content > 
<td bgcolor="#FFFFFF">&nbsp;{$user.id}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.user}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.address}</td> 
</tr> 
</volist> 
然后 创建 User 模块 文件 夹 ， 编 辑 insert 操作 的 模板 文件 mdex.html， 创 建 添加 用 户 信息 的 表单 。 其 关键 
代码 如 下 : 
<form method="post"” action=”_URL__/insert" > 


<table width="265" border="0" cellspacing="0" cellpadding="0"> 
<tr> 


<td class="title" id="td"> 用 户 名 : </td> 
<td><input name="user type="text" size="15" /></td> 
</tr> 
<tr> 
<td class="title" id="td"> 密 码 : </td> 
<td><input name="pass" type="password" size="15" /></td> 
</tr> 
<tr> 
<td class="title" id="td"> 地 址 : </td> 
<td><input name="address" type="text" size="20" /></td> 
</tr> 
</table> 
<input type="image" name="imageField" id="imageField" src=”_ROOT_/Publicimages/66_05.gif" /> 
</form> 

(7) 定位 到 Home\Conf 目录 ， 编 辑 前 台 项 目的 数据 库 配 置 文件 config.php。 

(8) 定位 到 Home\Lib\Action 目录 ， 创 建 前 台 项 目 控制 器 Index。 定 义 index 方法 查询 数据 库 中 的 用 户 
信息 ， 并 且 将 查询 结果 赋 给 模板 变量 ， 定 义 insert 方法 ， 通 过 R 快捷 方式 调用 Admin 项 目 UserAction 控制 
器 的 insert 操作 方法 完成 数据 的 添加 操作 。IndexAction.class.php 的 代码 如 下 : 

<?php 
header("Content-Type:text/html; charset=utf-8"); /设置 页 面 编码 格式 
class IndexAction extends Action{ 

public function index() { 


$db = new Model(user):; /实例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
$select = $db->select(); /查询 数据 
Sthis->assign('select', $select); /模板 变量 赋值 
$this->display(); /| 输出 模板 
外 
public function insert() { 
Sins=R("User","insert","Admin"); /远程 调用 Admin 项 目 UserAction 控制 器 的 insert 操作 方法 
Sins->Create(); /1 创建 数据 对 象 
$result = $ins->add(); // 写 入 数据 库 
} 
» 
?> 


(9) 定位 到 Home\Tpl\default 目录 下 ， 创 建 mdex 模块 文件 夹 ， 编 辑 index 操作 的 模板 文件 index html。 
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在 index.html 模板 文件 中 ， 创 建 表单 提交 用 户 注册 信息 ， 循 环 输出 模板 变量 传递 的 数据 。 
(10) 在 正 浏 览 器 中 输入 http:/127.0.0.1:82/TM/24/24.4/， 其 运行 效果 如 图 24.13 所 示 。 


后 台 管 理 系统 “| 
ne: Tet 
ET 
她 址 : 卡 圭 市 
避风 
[9 用 户 名 E22) 
mr 长 可 市 
mof 四 平市 
Te 长 覃 市 


图 24.13 ” 跨 模 块 调用 完成 用 户 注册 


， 


[| 
参数 根据 实际 应 用 决定 小 

redirectt[ 项 目 :/][ 路 由 @J[ 分 组 名 -模块 1 操作 ? 参数 1= 值 8 参数 N= 值 NJ) 

或 者 用 数组 的 方式 传 入 参数 : 

redirectt[ 项 目 :路 由 @][ 分 组 名 -模块 1 操作 ,array( 参 数 1=> 值 1 [参数 N=> 值 N]) 

如 果 不 定义 项 目 和 模块 的 话 ， 就 表示 当前 项 目 和 模块 名 称 。 例 如 : 

S$this->redirect('Index/index',", 5, 页面 跳 转 中 小 /页面 重 定向 

停留 5 秒 后 跳 转 到 Index 模块 的 index 操作 方法 ， 并 且 显 示 页 面 跳 转 中 字样 ， 重 定向 后 会 改变 当前 
的 URL 地 址 。 


24.5 ThinkPHP 的 模型 


句 4 视频 讲解 : 光盘 \TM\Video\ 第 24 章 \ThinkPHP 的 模型 .exe 
顾名思义 ， 模 型 就 是 按照 某 一 个 形状 进行 操作 的 代名词 。 模 型 的 主要 作用 是 ， 封 装 数据 库 的 相关 逻辑 。 
也 就 是 说 ， 每 执行 一 次 数据 库 操作 ， 都 要 遵循 定义 的 数据 模型 规则 来 完成 。 


24.5.1 ”模型 的 命名 


在 定义 模型 时 ，ThinkPHP 要 求 数据 库 的 表 名 和 模型 类 的 命名 遵循 一 定 的 规范 ， 首 先 数据 库 的 表 名 和 字 
段 全 部 采用 小 写 形式 ， 模 型 类 的 命名 规则 是 除去 表 前 缀 的 数据 表 名 称 ， 并 且 首 字 母 大 写 ， 然 后 加 上 模型 类 
的 后 缀 定义 。 

例如 ，UserModel 表示 User 数据 对 象 ，( 假 设 数 据 库 的 前 级 定义 是 think_〉 其 对 应 的 数据 表 应 该 是 
think_user; UserTypeModel 对 应 的 数据 表 是 think_user_type。 

如 果 你 的 规则 和 系统 的 约定 不 符合 ， 那 么 需要 设置 Model 类 的 tableName 属性 。 在 ThinkPHP 的 模型 里 
面 ， 有 两 个 数据 表 名 称 的 定义 。 

(1) tableName 不 包含 表 前 后 级 的 数据 表 名 称 ， 一 般 情况 下 默认 和 模型 名 称 相同 ， 只 有 当 表 名 和 当前 
的 模型 类 的 名 称 不 同 的 时 候 才 需 要 定义 。 例 如 ， 在 数据 库 里 面 有 一 个 think_categories 表 ， 而 定义 的 模型 类 
名 称 是 CategoryModel， 按 照 系统 的 约定 ， 这 个 模型 的 名 称 是 Category， 对 应 的 数据 表 名 称 应 该 是 


S14 


第 24 章 ThinkPHP 框架 


think category〈 全 部 小 写 ) ， 但 是 现在 的 数据 表 名 称 是 think categories， 因 此 就 需要 设置 tableName 属性 来 


改变 默认 的 规则 假设 已 经 在 配置 文件 里 面 定义 了 DB_PREFIX 为 think ) 。 
protected $tableName = 'categories'; 


(ot rr 


(2) trueTableName 包含 前 后 缀 的 数据 表 名 称 ， 也 就 是 数据 库 中 的 实际 表 名 ,该 名 称 无 需 设置 ,只 有 当 
上 面 的 规则 都 不 适用 的 情况 或 者 特殊 情况 下 才 需 要 设置 。 例如 ， 数据库 中 有 一 个 表 (top_depts) 的 前 级 和 其 
他 表 前 级 不 同 ， 不 是 think ”而 是 top ， 这 个 时 候 需要 定义 trueTableName 属性 。 

protected $trueTableName = 'top_depts'; 


WN 


和 5 注意 trueTableName 需要 完整 的 表 名 定义 。 


除了 数据 表 的 定义 外 ， 还 可 以 对 数据 库 进 行 定义 。 

dbName 定义 模型 当前 对 应 的 数据 库 名 称 ， 只 有 当前 的 模型 类 对 应 的 数据 库 名 称 和 配置 文件 不 同时 才 需 
要 定义 ， 例 如 。 

protected $dbName = 'top'; 


24.5.2 ”实例 化 模型 


在 ThinkPHP 2.0 版 本 中 ， 无 需 进行 任何 模型 定义 (只 有 在 需要 封装 单独 的 业务 逻辑 时 ， 模 型 类 才 是 必 
须 被 定义 的 ) ， 可 以 直接 进行 模型 的 实例 化 操作 。 根 据 不 同 的 模型 定义 ， 实 例 化 模型 的 方法 也 有 所 不 同 。 
下 面 来 分 析 一 下 什么 情况 下 使 用 什么 方法 。 

1. 实例 化 基础 模型 (Model) 类 

在 没有 定义 任何 模型 时 ， 可 以 使 用 下 面 的 方法 实例 化 一 个 模型 类 来 进行 操作 。 

$User = new Model(User); 

$User->select(); // 进 行 其 他 的 数据 操作 

或 者 使 用 M 快捷 方法 进行 实例 化 ， 其 效果 是 相同 的 。 

S$User = M('User'"); 

S$User->select(); /进行 其 他 的 数据 操作 

这 种 方法 最 简单 、 高 效 ， 因 为 不 需要 定义 任何 的 模型 类 ， 所 以 支持 跨 项 目 调用 。 缺 点 也 是 因为 没有 自 
定义 的 模型 类 ， 因 此 无 法 写 入 相关 的 业务 逻辑 ， 只 能 完成 基本 的 CURD 操作 。 在 例 24.2 和 例 24.4 中 采用 的 
都 是 实例 化 基础 模型 类 ， 对 数据 库 中 数据 进行 读 取 和 添加 操作 。 

2. 实例 化 其 他 模型 类 

第 一 种 方式 实例 化 因为 没有 模型 类 的 定义 ， 因 此 很 难 封装 一 些 额外 的 逻辑 方法 ， 不 过 大 多 数 情况 下 ， 
也 许 只 是 需要 扩展 一 些 通 用 的 逻辑 ， 那 么 就 可 以 尝试 下 面 一 种 方法 。 

M 方法 默认 是 实例 化 Model 类 ， 如 果 需 要 实例 化 其 他 模型 类 ， 可 以 使 用 : 

$User = M('User', 'CommonModel'); 

上 面 的 方法 等 效 于 : 

$User = new CommonModel(User); 

因为 系统 的 模型 类 都 能 够 自动 加 载 ， 因 此 不 需要 在 实例 化 之 前 手动 进行 类 库 导 入 操作 。 模 型 类 
commonModel 必须 继承 Model， 如 果 没 有 定义 别名 导入 的 话 ， 需 要 放 在 项 目 Model 下 。 我 们 可 以 在 
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CommonModel 类 里 面 定 义 一 些 通 用 的 逻辑 方法 ， 就 可 以 省 去 为 每 个 数据 表 定义 具体 的 模型 类 ， 如 果 项 目的 
数据 表 超过 100 个 ， 而 且 大 多 数 都 是 执行 基本 的 CURD 操作 ， 只 是 个 别 模型 有 一 些 复杂 的 业务 逻辑 需要 封 
装 ， 那 么 第 一 种 方式 和 第 二 种 方式 的 结合 是 一 个 不 错 的 选择 。 

3. 实例 化 用 户 定义 的 模型 (x Xx XModel) 类 

这 种 情况 是 使 用 最 多 的 ， 一 个 项 目 不 可 避免 地 需要 定义 自身 的 业务 逻辑 实现 ， 就 需要 针对 每 个 数据 表 
定义 一 个 模型 类 ， 如 UserModel、InfoModel 等 。 

定义 的 模型 类 通常 都 是 放 到 项 目的 Lib\Model 目录 下 面 。 例 如 : 

class UserModel extends Model{ 


Public function myfun(){ 
// 添 加 自己 的 业务 逻辑 
} 


} 

其 实 模型 类 还 可 以 继承 一 个 用 户 自 定义 的 公共 模型 类 ， 而 不 是 只 能 继承 Model 类 。 要 实例 化 自 定义 模 
型 类 ， 可 以 使 用 D 快捷 方法 ， 其 效果 是 相同 的 。 

S$User = DCUser); 

$User->select(); /进行 其 他 的 数据 操作 

D 方法 可 以 自动 检测 模型 类 , 不 存在 时 系统 会 抛 出 异常 同时 对 于 已 实例 化 过 的 模型 ,不 会 重复 去 实例 
化 。 默 认 的 D 方法 只 能 支持 调用 当前 项 目的 模型 ， 如 果 需 要 跨 项 目 调 用 ， 需 要 使 用 : 

S$User = D('User’, 'Admin'); /实例 化 Admin 项 目下 面 的 User 模型 

$User->select(); 

如 果 启 用 模块 分 组 功能 ， 还 可 以 使 用 : 

S$User = D(Admin.User); 

4. 实例 化 空 模型 类 

如 果 仅 仅 是 使 用 原生 SQL 查询 的 话 ， 不 需要 使 用 额外 的 模型 类 ， 实 例 化 一 个 空 模型 类 即 可 进行 操作 。 
例如 : 

$Model = new Model(); 

// 或 者 使 用 M 快捷 方法 实例 化 是 等 效 的 

/$Model = M(); 

$Model->query('SELECT * FROM think_user where status=1"); 

空 模型 类 也 支持 跨 项 目 调 用 。 

例 24.5 通过 M 方法 实例 化 Model 类 , 完成 数据 库 中 用 户 信息 和 类 别 信息 的 输出 。( 实 例 位 置 : 光盘 \TM\ 
Instance\24\24.5) 

其 关键 操作 步骤 如 下 。 

(1) 创建 TM 项 目 根 目录 ， 在 根 日 录 下 创建 项 目 文件 夹 App 和 Public 存储 CSS、 图 片 和 JS 脚本 等 
文件 。 
(2) 在 TM 项 目 根 目录 下 ， 编 辑 index.php 入 口 文 件 。 其 关键 代码 如 下 : 


<?php 

define('THINK_PATH", "../ThinkPHP/"); // 定 义 ThinkPHP 框架 路 径 〈 相 对 于 入 口 文件 ) 
define(APP_NAME'，TM"); /定义 项 目 名 称 

define(APP_PATH'，App'); /定义 项 目 路 径 
require(THINK_PATH."/ThinkPHP.php"); // 加 载 框架 入 口 文件 

App::run(); /实例 化 一 个 网 站 应 用 实例 


?> 

(3) 在 正 浏 览 器 中 运行 入 口 文件 ， 自 动 生成 项 目 目录 。 

(4) 定位 到 App\Conf 目录 下 ， 编 辑 config php 文件 ， 完 成 项 目 中 数据 库 的 配置 。 其 代码 如 下 : 
<?php 
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return array( 
'APP_DEBUG' => false, /关闭 调试 模式 
'DB_TYPE'=> 'mysql', /| 数据 库 类 型 
'DB_HOST'=> '|localhost', /数据 库 服务 器 地 址 
'DB_NAME'=>'db_database24', /| 数据 库 名 称 
'DB_USER'=>'root', /| 数据 库 用 户 名 
'DB_PWD'=>"111', /| 数据 库 密码 
'DB_PORT'=>'3306', /数据 库 端 口 
'DB_PREFIX'=>'think_', /| 数据 表 前 缀 

> 


(5) 定位 到 App\Lib\Action 目录 下 ,编写 项 目的 控制 器 。 创 建 Index 模块 ， 继 承 系统 的 Action 基础 类 ， 
定义 index0 方 法 ， 通 过 M 方法 实例 化 模型 类 ， 读 取 think_user 数据 表 中 的 数据 ， 并 且 将 查询 结果 赋 给 模板 
变量 ， 指 定 模板 页 ， 定义 type0 方 法 ， 通 过 M 方法 实例 化 模型 类 ， 读 取 类 型 数据 表 think type 中 的 数据 ， 同 
样 将 查询 结果 赋 给 模板 变量 ， 指 定 模板 页 。IndexAction.class.php 的 代码 如 下 : 

<?phI 
eh charset=utf-8"); // 设 置 页 面 编码 格式 
class IndexAction extends Action{ 

public function index(X{ 


$db = M('User'); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
$select = $db->select(); /查询 数据 
$this->assign('select,$select); /模板 变量 赋值 
S$this->display(); // 指 定 模板 页 
} 
public function type(X{ 
$dba = M('Type'); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
S$select = $dba->select(); /查询 数据 
$this->assign('select,$select); /模板 变量 赋值 
S$this->display('type'); // 指 定 模板 页 
} 
上 
?> 


(6) 定 位 到 App\Tpl\default 目录 下 ,创建 Index 模块 文件 夹 .首先 ,编辑 index 操作 的 模板 文件 index.html， 
循环 输出 模板 变量 传递 的 数据 。 其 关键 代码 如 下 : 
<volist name='select id='user > 
<tr class="content"> 
<td bgcolor="#FFFFFF">&nbsp;{$user.id}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.user}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.address}</td> 
</tr> 
</volist> 
然后 编辑 type.html 模板 文件 ， 循 环 输出 类 型 数据 表 中 的 数据 。 其 关键 代码 如 下 : 
<volist name='select id='type' > 
<tr class="content"> 
<td bgcolor="#FFFFFF">&nbsp:{$type.id}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$type.typename}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$type.dates}</td> 
</tr> 
</volist> 
(7) 在 正 浏 览 器 中 输入 http:/127.0.0.1:82/TM/24/24.5/， 其 运行 效果 如 图 24.14 所 示 。 在 正 浏览 器 中 


输入 http://127.0.0.1:82/TM/24/24.5/index.php/Index/type， 其 运行 效果 如 图 24.15 所 示 。 
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图 24.14 输出 用 户 信息 图 24.15 输出 类 别 信息 


peg 
em pr 
应 数据 表 的 字段 呢 ? 这 是 因为 ThinkPHP 可 以 在 运行 时 自动 获取 数据 表 的 字段 信息 ( 确切 地 说 ， 是 在 第 
一 次 运行 时 ， 将 其 存储 于 缓存 文件 ， 以 后 会 永久 缓存 字段 信息 ， 除 非 设置 不 缓存 或 者 出 除 ) ， 包 括 数据 
表 的 主键 字段 和 是 否 自动 增长 等 ， 如 果 需 要 显 式 获取 当前 数据 表 的 字段 信息 ， 可 以 使 用 模型 类 的 
getDbFields 方法 来 获取 。 如 果 在 开发 过 程 中 修改 了 数据 表 的 字段 信息 ， 需 要 清空 Data/_fields 目录 下 面 
的 缓存 文件 ， 让 系统 重新 获取 更 新 的 数据 表 字段 信息 


24.5.3 ”属性 访问 


ThinkPHP 利用 PHP 5 的 魔术 方法 机 制 来 实现 属性 的 直接 访问 。 这 也 是 最 常用 的 访问 方式 ， 通 过 数据 对 
象 访问 ， 例 如 : 


<?php 

S$User = new Model(User); 

S$User->find(1); 

echo $User->name; // 获 取 name 属性 的 值 
$User->name = 'ThinkPHP'; /设置 name 属性 的 值 
ea 

还 有 一 种 属性 的 操作 方式 是 通过 返回 数组 的 方式 。 例 如 : 

<?php 

$Type = D('Type'); /注意 这 里 返回 的 type 数据 是 一 个 数组 
$type = $Type->find(1); 

echo $type['name']; // 获 取 type 属性 的 值 
S$type[name'] = ThinkPHP'; /设置 type 属性 的 值 
?> 


24.5.4 ”连接 数据 库 


ThinkPHP 内 置 抽象 数据 库 访 问 层 ， 把 不 同 的 数据 库 操 作 封装 起 来 ， 只 需 使 用 公共 的 Db 类 进行 操作 ， 
而 无 需 针 对 不 同 的 数据 库 写 不 同 的 操作 代码 ，Db 类 会 自动 调用 相应 的 数据 库 适 配器 来 处 理 。 目 前 的 数据 库 
包括 Mysql、MsSQL、PgSQL、Sqlite、Oracle、Ibase 以 及 PDO 的 支持 ， 如 果 应 用 需要 使 用 数据 库 ， 必 须 配 
置 数据 库 连接 信息 ， 数 据 库 的 配置 文件 有 多 种 定义 方式 。 

(1) 在 项 目 配置 文件 里 面 定义 ， 在 前 面 的 实例 中 已 经 见识 过 了 。 其 代码 如 下 : 


<?php 

return array( 
'APP_DEBUG' => false, /关闭 调试 模式 
'DB_TYPE'=> 'mysql', /数据 库 类 型 
'DB_HOST'=> localhost'， /数据 库 服务 器 地 址 
'DB_NAME'=>'db_database24', /数据 库 名 称 
'DB_USER'=>'root', /| 数据 库 用 户 名 
'DB_PWD'=>"111', /| 数据 库 密码 
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'DB_PORT'=>'3306’, /数据 库 端口 
'DB_PREFIX'=>'think_', /| 数据 表 前 缀 
); 
?> 


系统 推荐 使 用 该 种 方式 ， 因 为 一 般 一 个 项 目的 数据 库 访 问 配置 是 相同 的 。 该 方法 系统 在 连接 数据 库 时 


会 自动 获取 ， 无 需 手动 连接 。 


of 
后 昌 可 以 对 每 个 项 目 定义 不 同 的 数据 库 连 接 信息 ,还 可 以 在 调试 配置 文件 里 面 定义 调试 数据 库 的 配 
置信 息 ， 如 果 在 项 目 配置 文件 和 调试 模式 配置 文件 里 面 同时 定义 了 数据 库 连 接 信息 ， 那 么 在 调试 模式 下 
面 后 者 生效 ， 部 署 模式 下 面前 者 生效 。 


置 : 


(2) 使 用 DSN 方式 在 初始 化 Db 类 时 传 参数 ， 代 码 如 下 : 
$db_dsn="mysql://root:111@127.0.0.1:3306/db_database24"; /定义 DSN 
$db = new Db(); /执行 类 的 实例 化 
$conn=$db->getlnstance($db_dsn); /连接 数据 库 ， 返 回 数据 库 驱 动 类 
该 方式 主要 用 于 在 控制 器 里 面 自己 手动 连接 数据 库 的 情况 ， 或 者 用 于 创建 多 个 数据 库 连接 。 
(3) 第 三 种 为 使 用 数组 传 参数 ， 代 码 如 下 : 
$dsn = array( 
"dbms' => 'mysql', 
"username' => "username'， 
"password' => 'password', 
'hostname' => 'localhost 
"hostport => '3306", 
"database' => 'dbname’ 


人 = new Db(); 

S$conn=$db->getInstance($dsn); /| 连接 数据 库 ， 返 回 数据 库 驱 动 类 

该 方式 用 于 手动 连接 数据 库 或 者 创建 多 个 数据 库 连 接 。 

例 24.6 通过 DNS 方式 和 数组 传 参 的 方式 完成 与 数据 库 的 连接 ， 并 且 输 出 数据 库 中 的 数据 。 (实例 位 
光盘 \TM\Instance\24\24.6) 

其 关键 操作 步骤 如 下 。 

本 示例 是 例 24.5 的 延伸 ， 仍 然 输出 数据 库 中 用 户 和 类 别 表 中 的 数据 ， 只 是 对 其 连接 数据 库 的 方法 进行 


了 修改 。 


(1) 删除 App\Conf 目录 下 的 配置 文件 config.php。 
(2) 在 App\LibWAction\Index 目录 下 ， 修 改 控制 器 Index。 在 index0) 方 法 中 ， 应 用 DNS 方式 完成 与 数 


据 库 的 连接 ， 并 且 查 询 think_user 表 中 的 数据 。 其 关键 代码 如 下 : 


public function index(X{ 
$db_dsn="mysql://root:111@127.0.0.1:3306/db_database24"; // 定 义 DSN 


$db = new Db(); /| 执行 类 的 实例 化 
$conn=$db->getinstance($db_dsn); /连接 数据 库 ， 返 回 数据 库 驱 动 类 
$select=$conn->query('select * from think_user); /| 执行 查询 语句 
S$this->assign('select', $select); // 模 板 变量 赋值 

Sthis->display(); /指定 模板 页 


(3) 在 type() 方 法 中 ， 应 用 数组 传递 参数 ， 完 成 数据 库 的 连接 操作 ， 并 且 查 询 think type 表 中 的 数据 。 


其 关键 代码 如 下 : 


public function type(){ 
$dsn = array( 


_ 国 
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"dbms' => 'mysql', 
"username' => Toot ， 
"password => '111 
"hostname' => ‘localhost', 
"hostport => '3306' 

"database' => 'db_database24' 


); 
$db = new Db(); 


$conn=$db->getinstance($dsn); /连接 数据 库 ， 返 回 数据 库 驱动 类 
$select=$conn->query('select * from think_type'); /执行 查询 语句 
Sthis->assign('select', $select); /模板 变量 赋值 
Sthis->display('type'); // 指 定 模板 页 


h 
上 述 是 在 例 24.5 中 所 做 的 修改 , 至 于 其 他 步骤 与 例 24.5 相同 , 这 里 不 再 袭 述 了 。 其 运行 结果 也 与 例 24.5 
相同 。 
(4) 在 模型 类 里 面 定义 参数 ， 连 接 数据 库 ， 代 码 如 下 : 
protected $connection = array( 
"dbms' => 'mysql', 
'Username' => 'username', 
"password' => 'password', 
'hostname' => 'localhost', 
"hostport => '3306', 
'database' => 'dbname’ 


) 

// 或 者 使 用 下 面 的 方式 定义 

protected $connection = "mysql://usermname:password@localhost:3306/DbName"; 

如 果 在 某 个 模型 类 里 面 定义 了 connection 属性 , 则 在 实例 化 模型 对 象 时 , 会 使 用 该 数据 库 连接 信息 进行 
数据 库 连 接 。 通 常用 于 某 些 数 据 表 位 于 当前 数据 库 连 接 之 外 的 其 他 数据 库 。 


YY 
入 胃 ThinkPHP 并 不 是 在 一 开始 就 会 连接 数据 库 ， 而 是 在 有 数据 查询 操作 时 才 会 去 连接 数据 库 。 特 
殊 的 情况 是 ， 在 系统 第 一 次 操作 模型 时 ， 框 架 会 自动 连接 数据 库 获取 相关 模型 类 的 数据 字段 信息 ， 并 缓 
存 下 来 。 


(5) 使 用 PDO 方式 连接 数据 库 。 这 里 在 项 目 配置 文件 中 ， 应 用 PDO 连接 数据 库 。 其 定义 的 数组 内 容 
如 下 : 
return array( 
'DB_TYPE'=> 'pdo', 
/注意 DSN 的 配置 针对 不 同 的 数据 库 有 所 区 别 
'DB_DSN'=> 'mysql:host=|localhost;dbname=db_database24', 
'DB_USER'=>'root', 
'DB_PWD'=>"111", 
'DB_PREFIX'=>'think_', 
// 其 他 项 目 配置 参数 
'APP_DEBUG' => false, /关闭 调试 模式 
); 


ot 在 使 用 PDO 方式 时 ， 要 注意 检查 你 的 PHP 环境 是 否 开启 相关 的 PDO 模块 。 同 时 还 要 确保 你 
的 ThinkPHP 核心 包 中 包含 DbPdo.class.php 文件 。 另外， 还 要 注意 参数 DB_DSN 仅 对 PDO 方式 连接 才 
有 效 。 
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例 24.7 在 项 目 配 置 文 件 中 ， 以 PDO 方式 连接 数据 库 ， 并 且 输 出 数据 库 中 的 数据 。〔 实 例 位 置 光盘 \ 
TM\24\24.7) 
我 们 知道 在 例 24.5 中 数据 库 的 连接 方法 定义 到 配置 文件 config.php 中 ， 而 应 用 PDO 连接 MySQL 数据 
库 仍然 需要 在 配置 文件 中 进行 操作 ， 那 么 我 们 只 需 对 例 24.5 中 的 config.php 文件 进行 修改 ， 就 完成 了 一 个 
新 的 实例 ， 应 用 PDO 连接 MySQL 数据 ， 并 且 输 出 查询 结果 。 其 修改 后 的 config.php 文件 的 代码 如 下 : 
<?php 
return array( 
'DB_TYPE'=> 'pdo', 
/注意 DSN 的 配置 针对 不 同 的 数据 库 有 所 区 别 
'DB_DSN'=> 'mysql:host=localhost;:dbname=db_database24', 
'DB_USER'=>'root', 


'DB_PWD'=>"111', 
'DB_PREFIX'=>'think_', 
…// 其 他 项 目 配置 参数 
'APP_DEBUG' => false, /关闭 调试 模式 

上 

本 例 的 运行 结果 与 24.5 相同 ， 这 里 不 再 袭 述 。 

24.5.5 ”创建 数据 
ThinkPHP 可 以 自动 根据 表单 数据 创建 数据 对 象 ， 这 个 优势 在 一 个 数据 表 的 字段 非常 之 多 的 情况 下 尤其 


明显 。 例 如， 在 User 控制 器 中 ， 定 义 insert0 方 法 ， 首 先 实例 化 模型 类 ， 然 后 调用 create0 方 法 根据 表单 提交 
的 POST 数据 创建 数据 对 象 ， 最 后 调用 add0 方 法 把 创建 的 数据 对 象 写 入 数据 库 。 其 关键 代码 如 下 : 


class UserAction extends Action{ // 定 义 类 ， 继 承 基础 类 
public function insert() { /定义 方法 
S$ins = new Model(user); /实例 化 模型 类 ， 传 递 参 数 为 没有 前 缀 的 数据 表 名 称 
S$ins->create(); /创建 数据 对 象 
$result = $ins->add(); // 写 入 数据 库 


Sthis->redirect('Index/index',", 5,' 页 面 跳 转 中 ');”，// 页 面 重 定向 
} 
} 
短 短 的 3 行 代码 ， 完 成 数据 的 添加 操作 。 其 具体 应 用 可 以 参考 例 24.4。 
其 中 的 create() 方 法 还 支持 其 他 方式 提交 的 数据 对 象 。 例 如 ， 以 数组 形式 提交 数据 ， 从 其 他 的 数据 对 象 
中 获取 的 数据 等 。 其 关键 代码 如 下 : 
// 数 组 形式 提交 数据 


$data['user] = 'mrsoft’; 
$data['address'] = ' 长 春 市 '; 


$User->create($data); 

/从 User 数据 对 象 创建 新 的 Member 数据 对 象 

S$User = M("User"); /实例 化 User 对 象 
S$User->find(1); // 读 取 数 据 
$Member = M("Member"); // 创 建 Member 对 象 
$Member->create($User); 


create( 方 法 在 创建 数据 对 象 的 同时 ， 还 实现 了 一 些 非常 有 意义 的 功能 ， 包 括 支持 多 种 数据 源 、 数 据 自 
动 验证 、 字 段 类 型 检查 和 数据 自动 完成 等 。 

create() 方 法 创建 的 数据 对 象 是 保存 在 内 存 中 ， 并 没有 实际 写 入 到 数据 库 中 ， 直 到 使 用 add0 或 者 saveO 
方法 ， 才 真正 将 数据 添加 到 数据 库 中 。 如 果 只 是 想 简单 创建 一 个 数据 对 象 ， 那么 可 以 使 用 data0 方 法 。 例如 ， 


_ 回 ) 
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实例 化 User 模型 ， 通 过 data0 和 add0 方 法 将 数据 添加 到 数据 库 中 。 其 关键 代码 如 下 : 
S$User = M(CUser); /实例 化 User 模型 
// 创 建 数据 后 写 入 到 数据 库 
$data['user] = 'mrsoft'; 
$data['address'] = ' 长 春 市 '; 
S$User->data($data)->add(); /| 执行 数据 对 象 的 创建 


人 注意 使 用 data() 方 法 创建 的 数据 对 象 不 会 进行 自动 验证 和 过 滤 操 作 ， 需 要 自行 处 理 。 但 在 进行 add 
或 者 save 操作 时 ， 数据 表 中 不 存在 的 字段 以 及 非法 的 数据 类 型 ( 例如 对 象 、 数 组 等 非 标量 数据 ) 是 会 自 
动 过 滤 的 ， 不 用 担心 非 数 据 表 字段 的 写 入 导致 SQL 错误 的 问题 。 


24.5.6 ”连贯 操作 


ThinkPHP 2.0 版 本 全 面 启用 模型 类 的 连贯 操作 方法 ， 可 以 有 效 地 提高 数据 存 取 的 代码 清晰 度 和 开发 效 
率 。 例 如 ， 查 询 一 个 User 表 的 满足 状态 为 1 的 前 5 条 记录 ， 并 按照 用 户 的 ID 排序 。 其 关键 代码 如 下 : 

S$User->where('status=1')->order('id')->limit(5)->select(); 

在 连贯 操作 中 ，select( 方 法 必须 放 到 最 后 一 个 ， 其 他 的 连贯 操作 方法 调用 顺序 没有 先后 。 如 果 不 习惯 使 
用 连贯 操作 ， 那 么 新 版 还 支持 直接 使 用 参数 进行 查询 的 方式 。 例 如 上 面 的 代码 可 以 改写 为 : 

S$User->select(array('order=>'id', where'=>'status=1", ‘imit'=>'5")); 

使 用 数组 参数 方式 的 话 ， 索 引 的 名 称 就 是 连贯 操作 的 方法 名 称 。 其 实 不 仅仅 是 查询 方法 可 以 使 用 连贯 
操作 ， 包 括 add、 save、delete 等 方法 都 可 以 使 用 。 例 如 : 

$User->where('id=1')->field('id,user,address')->find(); 

S$User->where('status=1 and id=1')->delete(); 


下 面 对 连 贯 操作 的 方法 进行 一 下 总 结 〈 更 多 的 用 法 将 在 CURD 操作 的 过 程 中 详细 描述 ) ， 如 表 24.3 


表 24.3 ”连贯 操作 方法 总 结 


方 法 名 描述 

where 用 于 查询 或 者 更 新 条 件 的 定义 。 参 数 支持 字符 串 、 数 组 和 对 象 

定义 要 操作 的 数据 表 名 称 。 可 以 动态 改变 当前 操作 的 数据 表 名 称 ， 需 要 写 数据 表 的 全 名 ， 包 含 前 级 ， 可 
以 使 用 别名 ， 例 如 : $Model->table('think_user user)->wWhere('status>1)->select(O: 

table0 方 法 的 参数 支持 字符 串 和 数组 ， 数 组 方式 的 用 法 : 

$Model->table(array('think_user=>'user','think group'=>'group'))->where('status>1")->selectO:; 

使 用 数组 方式 定义 的 优势 是 可 以 避免 因为 表 名 和 关键 字 冲 突 而 出 错 的 情况 。 如 果 不 定义 table0 方 法 ， 默 
认 会 自动 获取 当前 模型 对 应 或 者 定义 的 数据 表 

数据 对 象 赋值 。 可 以 用 于 新 增 或 者 保存 数据 之 前 的 数据 对 象 赋值 ， 例 如 : 

SModel->data($data)->addO: 

data $Model->data($data)->where('id=3")->save(): 

Data 方法 的 参数 支持 对 象 和 数组 ， 如 果 是 对 象 会 自动 转换 成 数组 。 如 果 不 定义 data0 方 法 赋值 ， 也 可 以 
使 用 create0 方 法 或 者 手动 给 数据 对 象 赋值 的 方式 

定义 要 查询 的 字段 。 参 数 支持 字符 串 和 数组 ， 例 如 : 

$Model->field('id.nickname as name')->selectO: 

$Model->field(array('id'.'nickname'=>'name’))->select|O: 

如 果 不 使 用 field0 方 法 指定 字段 的 话 ， 默 认 和 使 用 field(*") 等 效 


table 


field 
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续 表 
方 法 名 描 述 
对 结果 进行 排序 。 例 如 ，order(id desc) 
排序 方法 支持 对 多 个 字段 的 排序 ， 例 如 ，order('status desc.id asc) 
order0 方 法 的 参数 支持 字符 串 和 数组 ， 数 组 的 用 法 如 下 : 
order(array('status'=>'desc','id’)) 
结果 限制 。 在 ThinkPHP 中 ， 无 论 操 作 的 是 MySQL、MS SQL Server， 还 是 Oracle 数据 库 ， 其 limit 的 方 
limit 法 是 统一 的 ， 即 limit(offsetlength)。 例 如 ，limit(1.10)， 获 取 从 第 一 条 记录 开始 的 10 条 记录 。 
注意 ，limit(10) 与 limit(0.10) 是 等 效 的 
查询 分 页 。 属 于 新 增 特 性 ， 可 以 更 加 快速 地 进行 分 页 查询 。Page0 方 法 的 用 法 和 limit0 方 法 类 似 , 格式 为 : 
Page(page[.listRows]) 
Page 表示 当前 的 页 数 ，listRows 表示 每 页 显示 的 记录 数 。 例 如 ，Page(2.10)， 表 示 每 页 显示 10 条 记录 ， 
区 获取 第 2 页 的 数据 。 
listRow 如 果 不 写 的 话 ， 会 读 取 limit(length') 的 值 ， 例 如 ，limit(25)->page(3);， 表 示 每 页 显示 25 条 记录 ， 
获取 第 3 页 的 数据 。 如 果 limit 也 没有 设置 的 话 ， 则 默认 为 每 页 显示 30 条 记录 
Uy 查询 Group 支持 。 例 如 ，groupCuser id)，group0 方 法 的 参数 只 支持 字符 串 


order 


入 起 有 关上 述 方法 的 应 用 ， 将 在 后 面 的 CURD 操作 中 体现 ， 这 里 不 再 一 一 举例 。 


24.5.7 CURD 操作 


ThinkPHP 提供 了 灵活 和 方便 的 数据 操作 方法 ，CURD (创建 、 更 新 、 读 取 和 删除 ) 是 4 个 最 基本 的 数 
据 库 操作 。CURD 操作 通常 与 连贯 操作 配合 使 用 。 下 面 将 对 各 种 操作 的 使 用 方法 进行 分 析 〈 在 执行 类 的 实 
例 化 操作 时 ， 统 一 使 用 M 方法 ) 。 


1. 创建 操作 
在 ThinkPHP 中 使 用 add0 方 法 完成 数据 的 添加 操作 。 其 使 用 方法 如 下 : 
S$User = M("User"); /实例 化 User 对 象 


$data[rname'] = ThinkPHP' 

$data[email] = ThinkPHPQ@gmailcom'; 

$User->add($data); 

或 者 使 用 data0 方 法 进行 连贯 操作 。 其 代码 如 下 : 

$User->data($data)->add(); 

如 果 在 add 之 前 已 经 创建 数据 对 象 的 话 ( 如 使 用 了 create0 或 者 data0 方 法 ) ，add0 方 法 就 不 需要 再 传 入 
数据 了 。 

2. 读 取 数 据 

在 ThinkPHP 中 读 取 数据 的 方式 很 多 ， 通 常 分 为 读 取 某 个 字段 的 值 、 读 取 数 据 和 读 取 数 据 集 。 读 取 字 段 
的 值 使 用 getField0 方 法 ， 读 取 数 据 使 用 find0 方 法 ， 读 取 数 据 集 使 用 select0 方 法 。 

getField0 方 法 读 取 某 个 字段 的 值 ， 如 果 传 入 多 个 字段 的 话 ， 可 以 返回 一 个 关联 数组 。 返回 的 list 是 一 个 
数组 ， 键 名 是 用 户 的 id， 键 值 是 用 户 的 昵称 mickname。 例 如 ， 获 取 ID 为 3 的 用 户 的 昵称 和 获取 所 有 用 户 的 
ID 和 昵称 列表 。 


S$User = M("User"); /实例 化 User 对 象 
S$nickname = $User->where('id=3')->getField('nickname’); /获取 ID 为 3 的 用 户 的 昵称 
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Slist = $User->getField('id,nickname'); // 获 取 所 有 用 户 的 ID 和 昵称 列表 

select0 方 法 的 返回 值 是 一 个 二 维 数组 ， 如 果 没 有 查询 到 任何 结果 的 话 ， 也 是 返回 一 个 空 的 数组 。 配合 上 
面 提 到 的 连贯 操作 方法 可 以 完成 复杂 的 数据 查询 。 例如， 查找 status 值 为 1 的 用 户 数据 并 以 创建 时 间 排 序 返 
回 10 条 数据 。 

S$User = M("User"); /实例 化 User 对 象 

Slist = $User->where('status=1')->order('create_time')->limit(10)->select(); 

find0 方 法 与 select0 类 似 ，select 可 用 的 所 有 连贯 操作 方法 也 都 可 以 用 于 find0 方 法 ， 区 别 在 于 find0 方 
法 最 多 只 会 返回 一 条 记录 ， 因 此 limit0 方 法 对 于 find 查询 操作 是 无 效 的 。 例 如 ， 查 找 status 值 为 1、name 
值 为 think 的 用 户 数 据 。 

$User = M("User"); /实例 化 User 对 象 

$User->where('status=1 and name="think" ")->find(); 


et 


例 24.8 通过 add(0 方 法 向 数据 库 中 添加 数据 ， 然 后 ， 查 询 数据 表 中 用 户 名 等 于 mr 的 记录 ， 按 照 降 圭 
排列 ， 循 环 输出 3 条 记录 。〈 实 例 位置 ; 光盘 \TM\Instance\24\24.8) 

这 里 在 讲解 实例 的 实现 步骤 过 程 中 ， 省 略 了 项 目 目录 的 创建 、 入 口 文件 的 编写 和 配置 文件 的 设置 ， 其 
具体 步骤 可 以 参考 例 24.4 或 者 例 24.5。 这 里 将 直接 讲解 在 控制 器 中 如 何 完成 数据 的 添加 和 查询 操作 。 

其 关键 操作 步骤 如 下 。 

(1) 定位 到 24.8\App\Lib\Action\ 目 录 下 ， 编 写 项 目 控制 器 。 创 建 ndex 模块 ， 继 承 系统 的 Action 基础 
类 ， 定 义 index() 方 法 ， 通 过 M 方法 实例 化 模型 类 ， 应 用 连贯 操作 中 的 where0、order0、limit0 和 select0 方 
法 读 取 think_user 数据 表 中 的 数据 ， 并 且 将 查询 结果 赋 给 模板 变量 ， 指 定 模板 页 ; 定义 insert0 方 法 , 通过 M 
方法 实例 化 模型 类 ， 应 用 add0 方 法 向 指定 的 数据 表 中 添加 数据 。IndexAction.class.php 的 代码 如 下 : 

<7| 

on ap charset=utf-8"); // 设 置 页 面 编码 格式 


class IndexAction extends Action{ 
public function index(X{ 


$db = MOUser); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
$select = $db->where('user="mr")->order('id desc')->limit(3)->select(); 。 // 执 行 查询 语句 
S$this->assign('select', $select); /模板 变量 赋值 
$this->display(); /指定 模板 页 


上 
public function insert(){ 
$dba = M('User'); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
$data[user] = 'mr'; 
$data[pass] = md5(mrsoft); 
S$data['address'] = ' 长 春 市 '; 


Sresult=$dba->add($data); /执行 添加 数据 
if($resultX{ 
Sthis->redirect('Index/index',", 2,' 页 面 跳 转 中 '); // 页 面 重 定 向 
9) 
} 
由 
全 > 


2) 定位 到 App\Tpl\default 目录 下 ， 创 建 mdex 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 index.html， 
应 用 ThinkPHP 内 置 模板 引擎 中 的 foreach 标签 循环 输出 模板 变量 传递 的 数据 ， 创 建 添加 数据 的 表单 ， 将 数 
据 提交 到 控制 器 的 insert0 方 法 中 进行 处 理 。 其 关键 代码 如 下 : 


<foreach name='select item='user > 


@ 
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<tr class="content > 
<td bgcolor="#FFFFFF">&nbsp:{$user.id}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.user}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.address}</td> 
</tr> 
</foreach> 
<form id="form1" name="form1" method="post" action="_URL__/insert"> 
<input type="submit" name="button" id="button" value=" 数 据 添加 " /> 
</form> 


其 运行 效果 如 图 24.16 所 示 。 


ED E23 


过 一 数 据 江 加 之 后 、 


Sh 23 


图 24.16 数据 添加 、 查 询 后 的 运行 结果 
3. 更 新 数据 
在 ThinkPHP 中 使 用 save0 方 法 更 新 数据 库 ， 并 且 也 支持 连贯 操作 的 使 用 。 例 如 ， 更 新 数据 表 中 name 


和 email 字段 的 值 。 其 代码 如 下 : 


$User = M("User"); /实例 化 User 对 象 
$data[name'] = 'ThinkPHP"; // 要 修改 的 数据 对 象 属性 赋值 
$data[email] = 'ThinkPHP@gmail.com'; 

$User->where('id=5')->save($data); /根据 条 件 保存 修改 的 数据 


save0 方 法 在 执行 更 新 数据 的 操作 时 , 如 果 没 有 设置 任何 更 新 条 件 , 且 数 据 对 象 本 身 也 不 包含 主键 字段 ， 


那么 save 方法 不 会 更 新 任何 数据 库 的 记录 。 


第 二 种 方法 通过 data0 方 法 创建 要 更 新 的 数据 对 象 ， 然 后 通过 save0 方 法 进行 保存 。 例 如 : 


S$User = M("User"); /实例 化 User 对 象 

$data[name'] = 'ThinkPHP': // 要 修改 的 数据 对 象 属性 赋值 

$data[email] = 'ThinkPHP@gmail.com'; // 要 修改 的 数据 对 象 属性 赋值 

$User->where('id=5')->data($data)->save(); /根据 条 件 保 存 修改 的 数据 

第 三 种 方法 是 针对 某 个 字段 的 值 ， 应 用 setField0 方 法 进行 更 新 。 例 如 ， 更 新 数据 表 中 字段 name 的 值 ， 
条 件 是 ID 为 5 的 记录 。 

S$User = M("User"); /实例 化 User 对 象 

$User-> where('id=5"')->setField('name','ThinkPHP"); /更 改 用 户 的 name 值 

如 果 要 更 新 多 个 字段 的 值 ， 也 可 以 应 用 setField0 方 法 ， 只 需要 传 入 数组 即 可 ， 例 如 : 

S$User = M("User"); /实例 化 User 对 象 


/更改 用 户 的 name 和 email 的 值 
$User-> where('id=5)->setField(array(name',email),array(ThinkPHP',ThinkPHP@gmail.com )); 
第 四 种 ， 应 用 settne0 和 setDec( 方 法 对 统计 字段 〈 通 常 指 的 是 数字 类 型 ) 中 的 值 进行 增 减 操作 。 例 如 ， 


对 指定 用 户 的 积分 进行 增 、 减 操作 。 


S$User = M("User"); /实例 化 User 对 象 
S$User->setInc('score','id=5",3); /用 户 的 积分 加 3 
S$User->setInc('score','id=5"); /用 户 的 积分 加 1 
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S$User->setDec('score',id=5',5); /| 用户 的 积分 减 5 

S$User->setDec('score',id=5"); /用 户 的 积分 减 1 

4. 删除 数据 

在 ThinkPHP 中 使 用 delete0 方 法 删除 数据 库 中 的 记录 ， 同 样 可 以 使 用 连贯 操作 进行 删除 操作 。 例 如 ， 
删除 数据 表 中 id 为 5 的 记录 。 

S$User = M("User"); /实例 化 User 对 象 

$User->where('id=5")->delete(); /删除 id 为 5 的 用 户 数据 


delete0 方 法 可 以 用 于 删除 单个 或 者 多 个 数据 ， 主 要 取决 于 删除 条 件 ， 也 就 是 where0 方 法 的 参数 ， 也 可 
以 用 order0 和 limit0 方 法 来 限制 要 删除 的 个 数 。 例 如 ， 删 除 所 有 状态 为 0 的 5 个 用 户 数据 并 按照 创建 时 间 
排序 。 

S$User = M("User"); /实例 化 User 对 象 

$User->where('status=0")->order('create_time')->limit('5')->delete(); 

例 24.9 应 用 ThinkPHP 中 的 CURD 操作 ， 实 现 对 用 户 信息 的 查询 、 更 新 和 删除 操作 。〔 实 例 位 置 : 
光盘 \TM\Instance\24\24.9) 

这 里 直接 讲解 在 控制 器 中 如 何 定义 index0 方 法 完成 循环 输出 数据 库 中 的 数据 ; 定义 update0 方 法 完成 数 
据 的 更 新 ， 定 义 delete0 方 法 实现 数据 的 删除 。 

其 操作 步骤 如 下 。 

(1) 定位 到 App\Lib\Action\ 目 录 下 ， 编 写 项 目 控制 器 。 创 建 Index 模块 ， 继 承 系统 的 Action 基础 类 ， 
定义 index0 方 法 ， 以 记录 的 id 值 为 条 件 ， 降 暴 循 环 输出 10 条 记录 。 其 代码 如 下 : 

<?phl 

ro charset=utf-8"); // 设 置 页 面 编码 格式 


class IndexAction extends Action{ 
public function index(X{ 


$db = M('User'); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
$select = $db->order('id desc')->limit(10)->select(); 

S$this->assign('select', $select); /模板 变量 赋值 

Sthis->display(); /指定 模板 页 


} 
} 
定义 update0 方 法 ， 首 先 根据 超级 链接 传递 id 值 执 行 查询 ， 查 询 出 指定 的 数据 ， 并 且 将 查询 结果 赋 给 指 
定 的 模板 变量 。 然 后 ， 判 断 表单 提交 的 id 值 是 否 存在 ， 如 果 存 在 则 以 id 为 条 件 ， 对 指定 的 数据 进行 更 新 操 
作 。 其 关键 代码 如 下 : 
public function update(){ 


$db = M('User'); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
$select = $db->where('id=".$_GETT['id'])->select(); 
S$this->assign('select', $select); /模板 变量 赋值 
S$this->display(update); /指定 模板 页 
ifisset($_POST[id]) 人 
$data[user] = $_POST[user]:; // 要 修改 的 数据 对 象 属性 赋值 


$data[pass] = md5($_POST[pass]); 
$data[address] = $_POST[address'; 
$result=$db->where(id='.$_POST[id])->save($data); 。“”// 根 据 条 件 保 存 修改 的 数据 
if($resultX{ 

Sthis->redirect('Index/index',", 2, 数 据 更 新 成 功 ); 。“”// 页 面 重 定 向 
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定义 delete0 方 法 ， 根 据 超 链接 传递 的 id 值 ， 删 除数 据 库 中 指定 的 记录 。 其 关键 代码 如 下 : 
public function delete(){ 


$db = MCUser); /实例 化 模型 类 , 参数 数据 表 名 称 不 包含 前 缀 
$result=$db->where('id='$_GET[id])->delete(); /删除 id 为 5 的 用 户 数据 
if$result){ 
Sthis->redirect('Index/index',", 2,' 数 据 删除 成 功 '); // 页 面 重 定 向 
3 
F 
} 
> 


(2) 定位 到 App\Tpl\default 目录 下 ， 创 建 mdex 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 index.html， 
应 用 ThinkPHP 内 置 模板 引擎 中 的 foreach 标签 循环 输出 模板 变量 传递 的 数据 ;创建 更 新 和 删除 超 链接 ， 将 
指定 记录 的 id 作为 参数 进行 传递 。 其 关键 代码 如 下 : 
<foreach name='select item='user > 
<tr class="content"> 
<td bgcolor="#FFFFFF">&nbsp;{$user.id}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.user}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.address}</td> 
<td bgcolor="#FFFFFF"><a href="” URL__/update?id={$user.id}"> 更 新 </a>/<a href=" _URL__/delete?id= 
{$user.id}"> 删 除 </a></td> 
</tr> 
</foreach> 
(3) 在 Index 模块 文件 夹 下 ， 编 辑 update.html 模板 文件 ， 创 建 表单 ， 将 从 模板 变量 中 读 取 的 数据 作为 
表单 元 素 的 默认 值 进 行 输出 ， 用 表单 中 数据 提交 到 控制 器 的 update0 方 法 完成 数据 的 更 新 操作 。 其 关键 代码 
如 下 : 
<form id="form2" name="form2" method="post" action="”_URL__/update"> 
<table width="405" border="1" cellpadding="1" cellspacing="1" bgcolor="#99CC33" bordercolor="#FFFFFF"> 
<foreach name='select item='user > 
<tr class="content"> 
<td bgcolor="#FFFFFF" class="right" width="103"> 名 称 : </td> 
<td bgcolor="#FFFFFF" width="289"> <input type="hidden" name="id" id="hiddenField" value="{$user.id}” 
/><input name="user type="text" id="user" size="20" value="{$user.user}" /></td> 
</tr> 
<tr class="content"> 
<td bgcolor="#FFFFFF" class="right"> 密 码 : </td> 
<td bgcolor="#FFFFFF"><input name="pass" type="password" id="pass" size="20" value="{$user.pass}" /> 
</td> 
</tr> 
<tr class="content"> 
<td bgcolor="#FFFFFF" class="right">&nbsp; 地 址 : </td> 
<td bgcolor="#FFFFFF">&nbsp; 
<input name="address" type="text" id="address" size="30" value="{$user.address}" /> 
</td> 
</tr> 
<tr class="content"> 
<td bgcolor="#FFFFFF"><input type="submit" name="button" id="button" value=" 更 新 " /></td> 
</tr> 
</foreach> 
</table> 
</form> 


其 运行 效果 如 图 24.17 所 示 。 
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图 24.17 数据 更 新 和 删除 
24.6 ”ThinkPHP 的 视图 


名 0 视频 讲解 : 光盘 \TM\Video\ 第 24 章 \ThinkPHP 的 视图 .exe 

在 ThinkPHP 里 面 ,视图 由 两 个 部 分 组 成 : View 类 和 模板 文件 。Action 控制 器 直接 与 View 视图 类 进行 
交互 ， 把 要 输出 的 数据 通过 模板 变量 赋值 的 方式 传递 到 视图 类 ， 而 具体 的 输出 工作 则 交 由 View 视图 类 来 进 
行 ， 同 时 视图 类 还 完成 了 一 些 辅助 的 工作 ， 包 括 调用 模板 引擎 、 布 局 泻 染 、 输 出 替换 、 页 面 Trace 等 功能 。 
为 了 方便 使 用 ,在 Action 类 中 封装 了 View 类 的 一 些 输出 方法 ， 如 display、fetch 、assign 、trace 和 buildHtml 
等 方法 ， 这 些 方法 的 原型 都 在 View 视图 类 里 面 。 


24.6.1 模板 定义 


为 了 对 模板 文件 进行 更 加 有 效 的 管理 , ThinkPHP 对 模板 文件 进行 了 目录 划分 , 默认 的 模板 文件 定义 规则 是 : 

模板 目录 /模板 主题 /[ 分 组 名 /] 模 块 名 /操作 名 + 模板 后 缀 

模板 目录 默认 是 项 目下 面 的 Tpl， 模 板 主题 默认 是 default， 模 板 主题 功能 是 为 了 多 模板 切换 而 设计 的 ， 
如 果 有 多 个 模板 主题 ， 则 可 以 使 用 DEFAULT_THEME 参数 设置 默认 的 模板 主题 名 。 

在 每 个 模板 主题 下 面 ， 是 以 项 目的 模块 名 为 目录 ， 然 后 是 每 个 模块 的 具体 操作 模板 文件 ， 例 如 ，User 
模块 的 Add 操作 对 应 的 模板 文件 是 Tpl/default/User/add.html。 

模板 文件 的 默认 后 绥 是 .html， 后 绥 可 以 通过 TMPL TEMPLATE SUFFIX 来 配置 。 

如 果 项 目 启用 模块 分 组 功能 (假设 User 模块 属于 Home 分 组 ) ， 那 么 默认 对 应 的 模板 文件 就 会 发 生变 
化 ， 变 为 : Tpl/default/Home/User/add.html。 

分 组 功能 可 以 通过 TMPL_FILE_DEPR 参数 来 配置 ,进而 简化 模板 的 日 录 层 次 .例如 ,设置 TMPL FILE_ 
DEPR 等 于 “_”， 那 么 默认 的 模板 文件 就 变 成 : Tpl/default/Home/User_add.html。 


/ 
培 明 正 是 因为 系统 有 了 这 样 一 种 模板 文件 自动 识别 的 规则 ，display0 方 法 才 可 以 无 需 带 任何 参数 就 
输出 对 应 的 模板 。 


24.6.2 ”模板 赋值 
模板 赋值 是 在 Action 控制 器 中 完成 的 ， 通 过 assign0 方 法 将 控制 器 中 获取 的 数据 赋 给 模板 变量 。 例 如 : 


局 
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S$this->assign('name', $value); 
如 果 要 同时 输出 多 个 模板 变量 ， 可 以 使 用 数组 的 方式 进行 赋值 。 


$array = array(); 


$array[name] = ‘thinkphp’; 
$array[email] = "iu21st@gmail.com'; 
$array[phone] =  '12335678'; 
S$this->assign($array); 


这 样 ， 就 可 以 在 模板 文件 中 同时 输出 name、email 和 phone 3 个 变量 了 。 
24.6.3 ”指定 模板 文件 


模板 变量 赋值 后 就 需要 调用 模板 文件 来 输出 相关 的 变量 , 模板 调用 应 用 的 是 display0 方 法 。 下 面 讲解 如 
何 通 过 display0 方 法 完成 对 模板 的 调用 ， 如 表 24.4 所 示 。 
表 24.4 display() 方 法 的 应 用 


语法 格式 描述 

调用 当前 模块 的 其 他 操作 模板 。 例 如 ， 当 前 是 User 模块 下 面 的 read 操作 ， 而 要 
调用 User 模块 的 edit 操作 模板 ，$this->display(Cedit): 

调用 其 他 模块 的 操作 模板 ， 其 中 分 组 名 是 可 选 的 。 例 如 ， 当 前 是 User 模块 ， 要 
调用 Member 模块 的 read 操作 模板 ， 使 用 : S$this->display(Member:read"); 

如 果 要 调用 分 组 Admin 的 Member 模块 的 read 操作 模板 ， 使 用 : 

Sthis->display( Admin:Member:read'):; 

调用 其 他 主题 的 操作 模板 。 例 如 , 调用 Admin 主题 的 User 模块 的 edit 操作 模板 ， 


display( 操 作 名 ') 


display(' 分 组 名 :模块 名 :操作 名 ') 


人 isplay( 主 题名 @ 模 块 名 :操作 和 名) | 作用 ，Sthis->display(CAdmin@Useredit):。 此 种 方式 需要 指定 模块 和 操作 名 
直接 全 路 径 输出 模板 。 例 如 ， 直 接 输出 当前 的 Public 目录 下 面 的 menu.html 模板 
文件 , 使 用 : Sthis->display("./Public/menu.html"):。 这 种 方式 需要 指定 模板 路 径 和 
display( 异 板 文件 名 后 缀 , 这 里 的 Public 目录 是 位 于 当前 项 目 入 口 文 件 位 置 下 面 。 如 果 是 其 他 的 后 级 


文件 ， 也 支持 直接 输出 ， 例 如 ， S$this->display(./Public/menu.tp1); 只 
要 ./Public/menu.tpl 是 一 个 实际 存在 的 模板 文件 。 如 果 使 用 的 是 相对 路 径 的 话 , 要 
注意 当前 位 置 是 相对 于 项 目的 入 口 文 件 ， 而 不 是 模板 目录 
设置 模板 页 的 编码 。 例 如 ， 设 置 指定 模板 页 的 编码 为 gbk， 
Sthis->display(Member:read'. 'gbk’): 

display( 模板 文件 名 '，'charset. | 设置 指定 模板 文件 的 编码 和 格式 。 例 如 ， 设 置 模板 文件 为 utf-8 编码 ， 设 置 文件 
"format' 为 XML 格式 。 其 应 用 如 下 : Sthis->display(Member:read', utf-8", ‘text/xml); 


display( 模 板 文件 名 ''charset) 


p94 
这 在 第 二 种 用 法 中 ,不 需要 写 模板 文件 的 路 径 和 后 级 ， 严格 来 说 ,这 里 面 的 模块 名 和 操作 名 并 不 
一 定 需 要 有 对 应 的 模块 或 者 操作 ,只 是 一 个 目录 名 称 和 文件 名 称 而 已 。 例如 , 项目 中 可 能 没有 Public 模块 ， 
更 没有 Public 模块 的 menu 操作 ， 但 是 一 样 可 以 使 用 “S$this->display('Public:menu");” 语 句 输 出 这 个 模板 
SH 
模板 变量 赋值 后 , 在 指定 的 模板 文件 中 进行 输出 , 具体 的 输出 方法 需要 根据 选择 的 模板 引擎 来 决定 。 
如 果 使 用 的 是 内 置 的 模板 引擎 ， 请 参考 ThinkPHP 开发 完全 手册 模板 指南 中 的 内 容 ; 如 果 使 用 PHP 本 身 
作为 模板 引擎 ， 则 直接 在 模板 文件 里 面 输出 。 例 如 ，<?php echo $name.'['.$email.' '.$phone."]';?>。 
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24.6.4 ”特殊 字符 串 蔡 换 


在 进行 模板 输出 之 前 ， 系 统 还 会 对 模板 的 特殊 字符 串 进行 蔡 换 ， 实 现 模板 输出 的 替换 和 过 滤 。 这 个 机 
制 可 以 使 得 模板 文件 的 定义 更 加 方便 ， 默 认 的 蔡 换 规则 如 表 24.5 所 示 。 


表 24.5 ”模板 中 特殊 字符 串 的 替换 规则 


特殊 字符 串 蔡 换 描 述 
./Public 被 替换 成 当前 项 目的 公共 模板 目录 。 通 常 是 ， /项目 目录 /Tpl/default/Public/ 
PUBLIC 被 替换 成 当前 网 站 的 公共 目录 通常 是 ; /Public/ 
TMPL 蔡 换 成 项 目的 模板 目录 。 通 常 是 : /项 目 目录 /Tpldefault 
ROOT 会 蔡 换 成 当前 网 站 的 地 址 〈 不 含 域名 ) 
APP 替换 成 当前 项 目的 URL 地 址 (不 含 域名 ) 
URL 替换 成 当前 模块 的 URL 地 址 不 含 域名 ) 
ACTION 替换 成 当前 操作 的 URL 地 址 (不 含 域名 ) 
SELF 替换 成 当前 的 页 面 URL 


MA 
说 日 
[ws 
的 。 只 要 在 项 目 配置 文件 中 配置 TMPL PARSE _ STRING 就 可 以 完成 。 如 果 有 相同 的 数组 索引 ， 就 会 更 
改 系统 的 默认 规则 。 例 如 : 
TMPL_PARSE_STRING => array( 
' PUBLIC_' => /Common', // 更 改 默认 的 _PUBLIC_ 替换 规则 
'_UPLOAD_' => /Public/Uploads/", /| 增加 新 的 上 传 路 径 营 换 规则 
) 


例 24.10 ”实现 用 户 登录 功能 ， 将 登录 用 户 的 信息 存储 到 SESSION 变量 中 ， 应 用 ThinkPHP 中 提供 的 
分 页 扩展 类 和 Page 方法 完成 数据 的 分 页 输出 。〈 实 例 位 置 : 光盘 \TMNInstance\24\24.10) 
其 操作 步骤 如 下 。 
(1) 定位 到 App\Lib\Action\ 目 录 下 ， 编 写 项 目 控制 器 。 创 建 mdex 模块 ， 继 承 系统 的 Action 基础 类 ， 
定义 index0 方 法 , 验证 用 户 提交 的 用 户 名 和 密码 是 否 正确 , 如 果 正 确 则 将 登录 用 户 名 存储 到 session 变量 中 ， 
并 且 将 网 页 重 定向 到 main.html 页 面 。 其 代码 如 下 : 


<?php 
session_start(); /初始 化 session 变量 
header("Content-Type:text/html; charset=utf-8"); /设置 页 面 编码 格式 


class IndexAction extends Action{ 
public function index(X{ 
if(isset($_POST['user])X{ 
if(isset($_POST['user']) && isset($_POST[pass])){ 


$db = M(); // 实 例 化 模型 类 ,参数 数据 表 名 称 不 包含 前 缀 
S$select = $db->query("select * from think_user where user=".$_POST[user].” and 

pass=".$_POST[pass]""); /| 执行 查询 语句 ， 验 证 用 户 名 和 密码 是 否 正确 
if($select}{ 


$_SESSION[admin]=$_POST[user]; /将 登录 用 户 名 存储 到 session 中 
Sthis->redirect('Index/main',", 2,' 用 户 "$_POST[user].' 登录 成 功 ! ); // 页 面 重 定向 
}else{ 


@® 


第 24 章 “ThinkPHP 框架 


Sthis->redirect('"Index/index',", 2,' 用 户 名 或 者 密码 不 正确 ! '); // 页 面 重 定向 
j 
jelsef 
S$this->redirect('Index/index',", 2, 用 户 名 、 密 码 不 能 为 空 ! '); // 页 面 重 定向 
» 
Sthis->display(); 
} 
类 
?> 
定义 main() 方 法 ， 载 入 分 页 类 ， 完 成 数据 库 中 数据 的 分 页 查询 ， 并 且 将 查询 结果 赋 给 模板 变量 。 其 代码 
如 下 : 
public function main(){ 
$db = M('User'); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
// 进 行 分 页 数据 查询 ， 注 意 ，Page() 方 法 的 参数 的 前 面部 分 是 当前 的 页 数 ， 使 用 $_GETIp] 获 取 
if(isset($_GETT'p])X{ // 判 断 分 页 变量 是 否 存在 
$p=$_GET[p]; 
jelsef 
$p=1; 
} 
Slist = $db->where('address='…" 长 春 市 ")->order('id desc')->page($p.",1)->select(); /查询 数据 
S$this->assign('select', $list); /赋值 数据 集 
import("ORG.Uti.Page"); // 导 入 分 页 类 
$count = $db->where('address="." 长 春 市 ")->count(); /查询 满足 要 求 的 总 记录 数 
$Page = new Page($count,1); /实例 化 分 页 类 ， 传 入 总 记录 数 和 每 页 显示 的 记录 数 
$show = $Page->show(); // 分 页 显示 输出 
S$this->assign('page', $show); /赋值 分 页 输出 
$this->display(main); /| 输出 模板 


} 
定义 validatorcode0 方 法 ， 应 用 GD 库 中 的 函数 ， 根 据 超 链接 传递 的 值 生成 用 户 登 录 的 验证 码 。 其 代码 
如 下 : 
public function validatorcode(){ 


header('content-type:image/png'); /定义 标题 PNG 格式 图 像 
$im = imagecreate(65, 25); /定义 画布 

imagefill($im, 0, 0, imagecolorallocate($im, 200, 200, 200)); /区 域 填充 
S$validatorCode = $_GET[code']; // 获 取 提交 的 值 


imagestring($im, rand(3, 5), 10, 3, substr($validatorCode, 0, 1), imagecolorallocate($im, 0, rand(0, 
255), rand(0, 255))); 

imagestring($im, rand(3, 5), 25, 6, substr($validatorCode, 1, 1), imagecolorallocate($im, rand(0, 255), 
0, rand(0, 255))); 

imagestring($im, rand(3, 5), 36, 9, substr($validatorCode, 2, 1), imagecolorallocate($im, rand(0, 255), 
rand(0, 255), 0)); 

imagestring($im, rand(3, 5), 48, 12, substr($validatorCode, 3, 1), imagecolorallocate($im, 0, rand(0, 
255), rand(0, 255))); 

imagepng($im); /生成 PNG 图 像 

imagedestroy(); /| 销毁 图 像 

} 
(2) 定位 到 App\Tpl\default 目录 下 ， 创 建 mdex 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 index.html， 
载 入 CSS 样式 文件 和 JavaScript 文件 ， 创 建 表单 ， 完 成 用 户 登 录 信 息 的 提交 操作 。 其 关键 代码 如 下 : 
<link href="_ROOT__/Public/Css/style.css" rel="stylesheet" type="text/css" /> 
@ 


<js href=" _ROOT__/Public/Js/check:js" /> 
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<form name="form1" method="post"” action=”_URL_/index" onSubmit="return chkinput(this)" > 
<table width="265" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td class="title" id="td"> 用 户 名 : </td> 
<td><input name="user" type="text" size="15" /></td> 
</tr> 
<tr> 
<td class="title" id="td"> 密 码 : </td> 
<td><input name="pass" type="password" size="15" /></td> 
<ltr> 
<tr> 
<td class="title" id="td"> 验 证 码 : </td> 
<td> 
<input type="text" name="validatorCode" size="10" /> 
<input type="hidden" name="defValidatorCode" value="™" /> 
<script language="javascript"> 


var num1=Math.round(Math.random()*10000000); /| 生成 随机 数 
var num=num1.toString().substr(0,4); /截取 随机 数 的 前 4 个 字符 
document.write("<img name=codeimg src='"__URL__/validatorcode?code="+num+">"); 
// 将 截取 值 传 递 到 图 像 处 理 页 中 
form1.defValidatorCode.value=num; // 将 截取 值 赋 给 表单 中 的 隐藏 域 
function reCode()f /定义 方法 ,重新 生成 验证 码 
var num1=Math.round(Math.random()*10000000); /生成 随机 数 
var num=num1.toString().substr(0,4); // 截 取 随 机 数 
document.codeimg.src=" _URL__/validatorcode?code="+num; // 将 截取 值 传递 到 图 像 处 理 页 中 
form1.defValidatorCode.value=num; // 将 截取 值 赋 给 表单 中 的 隐藏 域 
Di 
</script> 
<a href="javascript:reCode()" class="content"> 看 不 清 </a> 
</td> 
</tr> 
</table> 
<input type="image" name="imageField" id="imageField" src=”_ROOT_/Publicimages/66_05.gif' /> 
</form> 


(3) 在 Index 模块 文件 夹 下， 编辑 main html 文件 ， 通 过 模板 引擎 中 的 session 标签 输出 当前 登录 的 用 
户 名 ， 通 过 foreach 标签 循环 输出 模板 变量 传递 的 数据 ， 最 后 输出 模板 变量 传递 的 分 页 超 链接 。 其 关键 代码 
如 下 : 
<table width="405" border="1" cellpadding="1" cellspacing="1" bgcolor="#99CC33" bordercolor="#FFFFFF"> 
<tr> 
<td colspan="3" bgcolor="#FFFFFF" class="title" align="center"> 当 前 登录 用 户 : {$Think.session.admin}</td> 
</tr> 
<foreach name='select item='user > 
<tr class="content > 
<td bgcolor="#FFFFFF">&nbsp;{$user.id}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.user}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.address}</td> 
</tr> 
</foreach> 
<tr class="content"> 
<td colspan="3" bgcolor="#FFFFFF">&nbsp;{$page}</td> 
</tr> 
</table> 


他 
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其 运行 效果 如 图 24.18 所 示 。 
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图 24.18 ”用户 登 录 和 数据 的 分 页 输出 


24.7 内 置 ThinkTemplate 模板 引擎 


区 il 视频 讲解 : 光盘 \TM\Video\ 第 24 章 \ 内 置 ThinkTemplate 模板 引擎 .exe 

ThinkPHP 内 置 了 一 个 基于 XML 的 性 能 卓越 的 模板 引擎 ThinkTemplate， 这 是 一 个 专门 为 ThinkPHP 服 
务 的 内 置 模板 引擎 。ThinkTemplate 是 一 个 使 用 XML 标签 库 技术 的 模板 引擎 , 支持 两 种 类 型 的 模板 标签 ( 普 
通 标签 和 XML 标签 ) ， 使 用 了 动态 编译 和 缓存 技术 ， 而 且 支持 自 定义 标签 库 。 

ThinkTemplate 模板 引擎 生成 的 编译 文件 默认 存储 于 Runtime/Cache 目录 下 面 , 以 模板 文件 的 MD5 编码 
作为 缓存 文件 名 保存 。 

下 面 介绍 一 些 ThinkTemplate 模板 引擎 中 的 常用 标签 ， 如 表 24.6 所 示 。 

表 24.6 ThinkTemplate 模板 引擎 中 的 常用 标签 

应 用 描述 


输出 模板 引擎 中 的 变量 。 注 意 模板 标签 的 “{” 和 “$” 之 间 不 能 有 任何 的 空 
格 ， 否 则 标签 无 效 


标签 名 称 
{$name} 


/| 输出 8$_SERVER 变量 
{$Think.server.script_name } 

// 输 出 $_SESSION 变量 
{$Think.session.session idlmds } 

/| 输出 8_GET 变量 
{$Think.get.pageNumber } 

// 输 出 $_COOKIE 变量 
{$Think.cookie.name } 

{$Think.now } // 现 在 时 间 
{$Think.templatelbasename } // 模 板 页 面 


系统 变量 。 除 了 常规 变量 的 输出 外 ， 模 板 引擎 还 支持 系统 变量 和 系统 常量 ， 
以 及 系统 特殊 变量 的 输出 。 它 们 的 输出 不 需要 事先 赋值 给 某 个 模板 变量 。 系 
统 变量 的 输出 必须 以 S$Think. 打头， 并且 仍然 支持 使 用 函数 


默认 值 输出 。 如 果 输 出 的 模板 变量 没有 值 ， 但 是 需要 在 显示 时 赋予 一 个 默认 
值 的 话 ， 可 以 使 用 default 语法 。 例 如 ，{Susernicknameldefault=" 明 日 科技 "} 
对 系统 变量 的 输出 也 可 以 支持 默认 值 ， 例 如 ，{SThink postnameldefault=" 名 
称 为 空 "} 


{$ 变 量 |default=" 默 认 值 "} 
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续 表 
标签 名 称 应 用 描述 
2 使 用 Include 标签 来 包含 外 部 的 模板 文件 

人 克 锌 用 后 完整 文件 名 的 包含 ， 例 如 ，<include file="./Tpl/default/Public/header.html" />。 
7 区 广 汪 时 轴 类 交大 全 如下 十 计 广 和 这 种 情况 下 ， 模 板 文件 名 必须 包含 后 级 。 使 用 完整 文件 名 包含 时 ， 特 别 要 注 
<include file=" 操 作 名 "人 意 文件 包含 指 的 是 服务 器 端 包含 ， 而 不 是 包含 一 个 URL 地址， 也 就 是 说 file 
// 包 含 其 他 模块 的 操作 模板 参数 的 写法 是 服务 器 端的 路 径 ， 如 果 使 用 相对 路 径 的 话 ， 是 基于 项 目的 入 口 
<include file=" 模 块 名 :操作 名 " /> 文件 位 置 

// 包 含 其 他 模板 主题 的 模块 操作 模板 用 变量 包含 。 例 如 ，<include file="$tpIName" />。 给 $tpIName 赋 不 同 的 值 就 
_include file-" 主 题名 @ 模 块 名 :操作 名 " 放 可 以 包含 不 同 的 模板 文件 ， 变 量 的 值 的 用 法 和 上 面 的 用 法 相同 。 

/用 变量 控制 要 导入 的 模板 注意 : 由 于 模板 解析 的 特点 ， 从 入 口 模板 开始 解析 ， 如 果 外 部 模板 有 所 更 改 ， 


<include file="$ 变 量 名 " /> 


<import type='js' file="Js.Util.Array" /> 
<import type='css' file="Css.common" /> 


<load href="../Public/Js/Common.js" /> 
<load href="../Public/Css/common.css" /> 


<js href="_ PUBLIC_ /Js/Common.js" /> 
<css href="../Public/Css/common.css" /> 


<volist name="list” id="vo" offset="5" 
length="10> 

{$vo.name} 

</volist> 


模板 引擎 并 不 会 重新 编译 模板 ， 除 非 缓存 已 经 过 期 。 如 果 修 改 了 包含 的 外 部 
模板 文件 后 ， 需 要 把 模块 的 缓存 目录 清空 ， 否 则 无 法 生效 

导入 文件 。 系 统 提供 专门 的 是 import 标签 和 load 标签 完成 文件 的 导入 操作 。 
第 一 个 是 import 标签 ， 导 入 方式 采用 类 似 ThinkPHP 的 import 函数 的 命名 
空间 方式 。import 标签 默认 的 起 始 路 径 是 网 站 的 Public 目录 ， 如 果 需 要 指定 
其 他 的 目录 ， 可 以 使 用 basepath 属性 ， 例 如 ，<import file="Js.Util.Array" 
basepath="./Common" /> 

第 二 个 是 load 标签 , 通过 文件 方式 导入 当前 项 目的 公共 JS 或 者 CSS。 在 href 
属性 中 可 以 使 用 特殊 模板 标签 普 换 ， 例 如 ，<load href="” PUBLIC _/Js/ 
Common.js" /> 

Load 标签 可 以 无 需 指 定 type 属性 ， 系 统 会 根据 后 级 自动 判断 。 

系统 还 提供 了 两 个 标签 别名 js 和 css 用 法 和 load 一致 

volist 标签 主要 用 于 在 模板 中 循环 输出 数据 集 或 者 多 维 数 组 。 标 签 参数 如 下 。 
name: 表示 模板 赋值 的 变量 名 称 ， 因 此 不 可 随意 在 模板 文件 中 改变 。 

id: 表示 当前 的 循环 变量 ， 可 以 随意 指定 ， 但 确保 不 要 和 name 属性 冲突 。 
支持 输出 部 分 数据 ， 例 如 输出 其 中 的 第 5 一 15 条 记录 。 

offset， 表 示 记 录 的 起 始 位 置 。 

length: 表示 记录 的 长 度 。 

mod: 控制 输出 记录 的 奇偶 性 ， 还 可 以 控制 在 指定 的 记录 换行 。 例 如 : 

// 输 出 偶数 记录 

<volist name="list" id="vo" mod="2" > 

<eq name="mod" value="1">{$vo.name}</eq> 

</volist> 

// 控 制 一 定 记录 的 换行 

<volist name="list" id="vo" mod="5" > 

{$vo.name} 

<eq name="mod" value="4"><br/></eq> 

</volist> 


<foreach name="list" item="vo" > 
{$vo.id} 

{$vo.name} 

</foreach> 


foreach 标签 用 于 循环 输出 ， 它 比 volist 标签 简洁 ， 没 有 volist 标签 那么 多 的 
功能 。 其 优势 是 可 以 对 对 象 进行 遍历 输出 ， 而 volist 标签 通常 是 用 于 输出 
数组 
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续 表 
应 用 描述 
switch 标签 ， 类 似 于 PHP 中 的 switch 语句 。 
其 中 name 属性 可 以 使 用 函数 以 及 系统 变量 ， 例 如 : 
<switch name="Think.get.userIdlabs"> 
<case value="1">admin</case> 


标签 名 称 


对 于 case 的 value 属性 可 以 支持 多 个 条 件 的 判断 ,使 用 “|” 进 行 分 割 , 例如 : 
<switch name="Think.get.type"> 
<case value="giflpnglipg"> 图 像 格式 </case> 


<switch name=" 变 量 " > 
<case value=" 值 1"> 输 出 内 容 1</case> 


<case value=" 值 2"> 输 出 内 容 2</case> ”|<default > 其 他 格式 
<default/> 默 认 情 况 </switch> 
</switch> 表示 如 果 $_GET["type"] 是 gif、png 或 者 jpg 的 话 ， 就 判断 为 图 像 格式 。 


也 可 以 对 case 的 value 属性 使 用 变量 ， 例 如 : 
<switch name="User.userId"> 

<case value="$adminId">admin</case> 

<case value="$memberId">member</case> 
<default />default 

</switch> 


使 用 变量 方式 的 情况 下 ， 不 再 支持 多 个 条 件 的 同时 判断 


例 24.11 仍然 以 用 户 登 录 和 数据 的 输出 为 背景 ， 应 用 ThinkPHP 中 提供 的 验证 码 类 和 分 页 类 生成 验证 
码 ， 完 成 数据 的 分 页 输出 。〈 实 例 位 置 ， 光盘 \TMNInstance\24\24.11) 
其 操作 步骤 如 下 。 
(1) 定位 到 App\LibWction\ 目 录 下 ， 编 写 项 目 控制 器 。 创 建 mdex 模块 ， 继 承 系统 的 Action 基础 类 ， 
定义 index0 方 法 ， 验 证 session 变量 存储 的 验证 码 与 用 户 提交 的 验证 码 是 否 相 同 ， 验 证 用 户 提交 的 用 户 名 和 
密码 是 否 正 确 ， 如 果 正 确 则 将 登录 用 户 名 存储 到 session 变量 中 ,并 且 将 网 页 重 定向 到 main.html 页 面 。 其 代 
码 如 下 : 
<?php 
session_start(); 
header("Content-Type:text/html; charset=utf-8"); /设置 页 面 编码 格式 
class IndexAction extends Action{ 
public function index(X{ 
if(isset($_POST['user])X{ 
if(isset($_POST[user') && isset($_POST[pass]) && isset($_POST['validatorCode'])}{ 
if($_SESSION[verify] == md5($_POST[validatorcode])){ /验证 验证 码 是 否 正确 
$db = M(); // 实 例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
S$select = $db->query("select * from think_user where user=".$_POST[user]." and 
pass=".$_POST[pass].""); 。 // 执 行 查询 语句 ， 验 证 用 户 名 和 密码 是 否 正确 
if($selectX{ 
$_ SESSION[admin]=$_POST[user]; 
$this->redirect(Index/main',", 2, 用 户 '.$_POST[user].' 登录 成 功 ! ); /页 面 重 定向 


}else{ 
S$this->redirect('IndeX/index,", 2, 用 户 名 或 者 密码 不 正确 ! '"); /页面 重 定向 
} 
se{ 
Sthis->redirect('Index/index',", 2, 验 证 码 不 正确 ! '); // 页 面 重 定向 


四 
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起 
}else{ 
Sthis->redirect('"Index/index',", 2, 用 户 名 、 密 码 不 能 为 空 ! "); // 页 面 重 定向 
} 
} 
Sthis->display(); 
} 
定义 main0 方 法 ， 载 入 分 页 类 ， 完 成 数据 库 中 数据 的 分 页 查询 ， 并 且 将 查询 结果 赋 给 模板 变量 。 这 里 应 
用 的 是 Page 类 和 limit 方法 完成 数据 的 分 页 输出 。 其 代码 如 下 : 
public function main(){ 


$db = M('User'); /实例 化 模型 类 ， 参 数 数据 表 名 称 不 包含 前 缀 
import("ORG.Util.Page"); /导入 分 页 类 

$count = $db->count(); /统计 总 记录 数 

/count = $User->where("status=1")->count(); // 查 询 满足 要 求 的 总 记录 数 

$Page = new Page($count,1); // 实 例 化 分 页 类 ， 传 入 总 记录 数 和 每 页 显示 的 记录 数 
S$show = $Page->show(); // 分 页 显示 输出 


// 进 行 分 页 数据 查询 ， 注 意 ，limit() 方 法 的 参数 要 使 用 Page 类 的 属性 
Slist = $db->order('id')->limit($Page->firstRow.','.$Page->listRows)->select(); 
集 


S$this->assign('select ,$list); // 赋 值 数 据 : 
S$this->assign('page', $show); // 赋 值 分 页 输出 
$this->display(main); // 输 出 模板 
定义 verify0 方 法 ， 载 入 ThinkPHP 中 提供 的 验证 码 扩展 类 ， 调 用 buildImageVerify0 方 法 生成 验证 码 。 
其 代码 如 下 : 
public function verify(){ 
import("ORG.Util.Image"); // 载 入 验证 码 类 


image::buildimageVerify(4,5,'png','65','30','verify'); ” // 生 成 验证 码 


NGC 
buildImageVerify (方法 的 语法 如 下 : 
buildimageVerify($length, $mode, $type, $width, $height, $verifyName) 
参数 说 明 如 表 24.7 所 示 。 


表 24.7 ”验证 码 类 中 buildimageVerify() 方 法 说 明 


验证 码 的 长 度 ， 默 认为 4 位 数 
验证 字符 串 的 类 型 ， 默 认为 数字 ， 其 他 支持 类 型 有 0 字母 1 数字 2 大 写字 母 3 小 写字 母 4 中文 5 


mode 混合 (去掉 了 容易 混淆 的 字符 o00L1 和 数字 01) 
type | 验证 码 的 图 片 类 型 ， 默 认为 png 
width | 验证 码 的 宽度 ， 默 认 会 自动 根据 验证 码 长 度 自动 计算 


height 。 ”| 验证 码 的 高 度 ， 默 认为 22 


verifyName | 验证 码 的 SESSION 记录 名 称 ， 默 认为 verify 


生成 验证 码 之 后 ， 需 要 在 模板 页 中 通过 <img src=" APP /Index/verify/" /> 输出 生成 的 验证 码 图 像 。 
在 控制 器 中 通过 如 下 代码 : 
if($_SESSION[verify] != md5($_POST[verify]){ 
S$this->error( 验 证 码 错 误 ! '); 


} 


人 后 
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(2) 定位 到 App\Tpl\default 目录 下 ， 创 建 mdex 模块 文件 夹 。 编 辑 index 操作 的 模板 文件 index.html， 
载 入 CSS 样式 文件 和 JavaScript 文件 ， 创 建 表单 ， 完 成 用 户 登录 信息 的 提交 操作 ， 通 过 img 标签 输出 生成 
的 验证 码 。 其 关键 代码 如 下 : 

<link href="_ROOT_/Public/Css/style.css" rel="stylesheet" type="text/css" /> 
<js href=" _ROOT_ /Public/Js/check.js" /> 
<form name="form1" method="post” action=”_URL_/index" onSubmit="return chkinput(this)" > 
<table width="265" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td class="title" id="td"> 用 户 名 : </td> 
<td><input name="user type="text" size="15" /></td> 
</tr> 
<tr> 
<td class="title" id="td"> 密 码 : </td> 
<td><input name="pass" type="password" size="15" /></td> 
</tr> 
<tr> 
<td class="title" id="td"> 验 证 码 : </td> 
<td><input type="text" name="validatorCode" size="10" /></td> 
<td><img src="__APP__/Index/verify/" /></td> 
</tr> 
</table> 
</form> 
(3) 在 Index 模块 文件 夹 下 ， 编 辑 main.html 文件 ， 通 过 模板 引擎 中 的 session 标签 输出 当前 登录 的 用 户 
名 ， 通 过 foreach 标签 循环 输出 模板 变量 传递 的 数据 ， 最 后 输出 模板 变量 传递 的 分 页 超 链接 。 其 关键 代码 如 下 : 
<table width="405" border="1" cellpadding="1" cellspacing="1" bgcolor="#99CC33" bordercolor="#FFFFFF"> 
<tr> 
<td colspan="3" bgcolor="#FFFFFF" class="title" align="center"> 当 前 登录 用 户 : {$Think.session.admin}</td> 
</tr> 
<foreach name='select item='user > 
<tr class="content"> 
<td bgcolor="#FFFFFF">&nbsp;{$user.id}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.user}</td> 
<td bgcolor="#FFFFFF">&nbsp;{$user.address}</td> 
</tr> 
</foreach> 
<tr class="content"> 
<td colspan="3" bgcolor="#FFFFFF">&nbsp;{$page}</td> 
</tr> 
</table> 


Ne 
B 
说 明 这 里 应 用 的 验证 码 是 区 分 字母 的 大 小 写 的 。 


其 运行 效果 如 图 24.19 所 示 。 
| - 登 
录 
成 
功 
配 司 
当前 登录 用 户 - mrsef 
LT 
四 名 称 搜 进 
1 Ls 卡 者 市 
和 记录 6 页 下 = 再 1 之 主 业 了 下 潭 唱 所 -而 


图 24.19 ”验证 码 类 和 分 页 类 的 应 用 效果 


PHP+MySQL 开发 实战 


24.8 小 结 


本 章 主要 介绍 ThinkPHP 框架 的 下 载 、 架 构 、 配 置 、 控 制 器 、 模 型 以 及 视图 。 并 通过 实例 ， 对 ThinkPHP 
的 各 种 应 用 进行 了 讲解 , 以 此 来 增加 读者 对 ThinkPHP 的 理解 .希望 通过 本 章 的 学 习 ,读者 能 够 掌握 ThinkPHP 
技术 ， 将 其 灵活 运用 到 实际 的 网 站 开发 中 。 


24.9 ”学 习 成 果 检 验 


1. 通过 ThinkPHP 中 的 扩展 类 生成 中 文 验证 码 ， 其 关键 是 载 入 ThinkPHP\Lib\ORG\UtiN\mage.class.php 
中 的 Image 类 ， 调 用 其 中 的 GBVerify0 方 法 ， 生 成 中 文 验 证 码 。 其 运行 效果 如 图 24.20 所 示 。〔 实 例 位 置 : 
光盘 \TM\Instance\24\24.12) 


NN 
注意 在 ThinkPHP 2.0 版 本 提供 的 Image 类 中 ， 在 定义 GBVerify0 方 法 时 有 一 处 错误 ， 需 要 读者 手 
动 修改 。 其 中 “$codex = msubstr($code, $i, 1);” 对 字符 串 进行 截取 调用 的 msubstr0 方 法 不 存在 。 正确 的 
调用 方法 是 “$codex = String::msubstr($code, $i, 1);”。 


2. 应 用 ThinkPHP 中 的 文件 上 传 扩展 类 实现 文件 上 传 的 功能 ， 上 传 JPG、GIF 或 者 PNG 格式 的 图 片 到 
服务 器 指定 的 文件 夹 下 ， 其 运行 效果 如 图 24.21 所 示 。 (实例 位 置 ， 光盘 \TM\Instance\24\24.13) 


用 户 加: mmr 
Ee [ee 
DE 
文件 上 倘 


一 选择 文件 。 EVAppSerAwwwATM CE [ 强 均 | 
24.20 ”中 文 验证 码 图 24.21 应 用 ThinkPHP 中 的 扩展 类 上 传 文件 


AE 


Zend Framework 框架 
( 凤 视频 讲解 : 39 分 钟 ) 


Zend Framework (简称 ZF) 是 由 Zend 公司 支持 开发 的 完全 基于 
PHP 5 的 开源 框架 ， 采用 MVC 架构 模式 来 分 离 应 用 程序 中 不 同 的 部 
分 ， 便 于 程序 的 开发 和 维护 。 其 拥有 丰富 的 组 件 支持 ， 模 块 化 的 结构 
设计 ， 易 于 扩展 和 完善 的 文档 资料 以 及 灵活 的 架构 设计 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

WH 了 解 Zend Framework 的 MVC 

WH 熟悉 Zend Framework 的 MVC 环境 搭建 

WI 掌握 Zend_Auth 身份 认证 

m 掌握 Zend_Db 数据 库 操作 

Wm 掌握 Zend_File 文件 控制 

Wm 掌握 Zend_Layout 网 站 布局 

让 掌握 Zend_Paginator 分 页 
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25.1 Zend Framework 的 MVC 介绍 


25.1.1 Zend Framework 概述 


Zend Framework 框架 由 Zend 公司 主创 ， 同 时 Google、Microsoft 和 StrikeIron 作为 合作 伙伴 为 Zend 
Framework 提供 了 大 量 技术 和 Web 服务 接口 。 

Zend Framework 中 的 组 件 都 是 独立 的 ， 都 不 依赖 于 其 他 组 件 ， 这 样 的 松 藉 合 结构 可 以 让 开发 者 独立 使 
用 组 件 。 

Zend Framework 版 本 更 新 速度 非常 快 ， 所 包含 的 组 件 也 不 断 增 加 ， 所 以 说 Zend Framework 框架 可 以 让 
一 个 从 事 PHP 工作 的 人 始终 保持 与 最 新 技术 的 接触 和 学 习 。 

Zend Framework 有 自己 成 熟 的 社区 ， 关 于 Zend Framework 的 任何 问题 都 能 在 社区 里 找到 答案 。 


25.1.2 Zend Framework 常用 组 件 


由 于 Zend Framework 的 组 件 都 是 非常 独立 的 〈 这 种 情况 称 为 松 耦合 ) ， 所 以 在 开发 Web 程序 或 者 软 
件 时 可 以 独立 使 用 这 些 组 件 的 组 合 。 表 25.1 列举 了 Zend Framework 的 常用 组 件 及 功能 。 


表 25.1 Zend Framework 常用 组 件 及 功能 


组 件 名 称 功 能 
Zend Acl 权限 控制 
Zend_Auth 主要 用 于 认证 
Zend Cache 为 应 用 程序 提供 缓存 支持 

Zend_Config 应 用 程序 的 配置 数据 参数 
Zend Db 提供 Zend Framework 与 MySQL 的 数据 库 操作 方式 
Zend File 开发 者 控制 文件 的 上 传 和 下 载 
Zend Layout 实现 应 用 程序 的 视图 布局 
Zend_Mail 实现 Zend Framework 发 送 和 接收 Email 
Zend_Paginator 提供 数据 的 分 页 显示 
Zend_Registry 提供 对 象 注册 表 ， 相 当 于 全 局 变量 在 程序 中 的 使 用 
Zend_Session 提供 Zend Framework 的 session 控制 
Zend_Validate 提供 一 组 通用 校 验 器 


25.1.3 ”MVC 原理 


MVC 是 一 种 经 典 的 程序 设计 理念 ， 它 将 应 用 程序 分 为 3 部 分 : 模型 层 (Model) 、 视 图 层 (View) 和 
控制 层 《Controller) ，MVC 即 是 这 3 部 分 英文 名 称 字母 的 缩写 。 

MVC 设计 模式 产生 的 原因 如 下 : 

应 用 程序 中 用 来 完成 任务 的 代码 一 一 模型 层 〈 也 叫 业 务 逻 辑 ) ， 通 常 是 程序 中 相对 稳定 的 部 分 ， 重 用 
率 高 。 而 与 用 户 交 互 界面 一 一 视图 层 ， 却 经 常 改 变 。 如 果 因 需求 变动 而 不 得 不 对 业务 逻辑 代码 进行 修改 ， 
或 者 要 在 不 同 的 模块 中 应 用 相同 的 功能 而 要 重复 编写 业务 逻辑 代码 ， 不 仅 降低 整体 程序 的 开发 进度 ， 也 会 
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使 未 来 的 维护 变 得 非常 困难 。 因 此 ， 将 业务 逻辑 代码 与 外 观 分 离 ， 能 够 更 方便 地 根据 需求 改进 程序 ， 这 就 
是 MVC 设计 模式 。 
在 PHP Web 开发 中 ，MVC 设计 模式 的 各 自 功能 及 相互 关系 如 图 25.1 所 示 。 


Controller 
(控制 层 ) 


图 25.1 MVC 关系 图 


模型 层 (Model) 

模型 层 是 应 用 程序 的 核心 部 分 ， 它 可 以 是 一 个 实体 对 象 或 一 种 业务 逻辑 ， 之 所 以 称 为 模型 ， 是 因为 它 
在 应 用 程序 中 有 更 好 的 重用 性 和 扩展 性 。 

视图 层 (View) 

视图 层 提供 应 用 程序 与 用 户 之 间 的 交互 界面 ,在 MVC 理论 中 ,这 一 层 并 不 包含 任何 业务 逻辑 , 仅 提 供 
-种 与 用 户 交 互 的 视图 。 

控制 层 (Controller) 

控制 层 用 于 对 程序 中 的 请 求 进行 控制 ， 其 作用 就 像 国 家 的 宏观 调控 ， 它 可 以 选择 调用 哪些 视图 或 者 模型 。 


25.2 ”Zend Framework 的 MVC 环境 搭建 


对 4 视频 讲解 : 光盘 \TM\Video\ 第 25 章 \Zend Framework 的 MVC 环境 搭建 .exe 
利用 Zend Framework 开发 项 目 ， 首 先 必须 保证 PHP 运行 环境 对 Zend Framework 框架 的 支持 。Zend 
Framework 环境 的 搭建 主要 根据 PHP 环境 而 来 ， 所 以 必须 搭建 Apache 和 MySQL。 


25.2.1 环境 配置 


搭建 Apache 和 MySQL， 搭 建 的 Apache 中 必须 有 PDO_MYSQL 模块 ， 如 果 安 装 环境 中 没有 此 模块 ， 
可 以 去 PHP 官方 网 站 中 下 载 。 
1. 配置 HTTPD.CONF 
(1) 进入 Apache 的 conf 目录 下 ， 使 用 编辑 工具 打开 httpd.conf 文件 ， 定 位 到 如 下 内 容 : 
#LoadModule rewrite_module modules/mod_rewrite.so 
(2) 该 句 中 的 “#” 符 号 用 于 在 .conf 文件 中 注释 掉 该 行 。 去 掉 “#” 符 号 后 ， 表 示 加 载 mod_rewrite 模块 。 
(3) 在 Apache 加 载 mod_rewrite.so 模块 后 需要 指定 生效 的 目录 ， 在 httpd.conf 文件 中 找到 所 有 为 
“AllowOverride None” 的 字符 串 ， 将 其 修改 为 “AllowOveride All”， 然 后 保存 httpd.conf 文件 即 可 开启 


mod rewrite 功能 。 


PHP+MySQL 开发 实战 


\ 


| pa ,a i i A 
从 全 文中 查找 “AllowOverride None”。 


2. 配置 PHP.INI 

Zend Framework 操作 MySQL 使 用 PHP 自 带 的 PDO_MYSQL 模块 .默认 的 PHP 是 不 开启 PDO_MYSQL 
模块 的 ， 所 以 必须 重新 对 PHP 环境 进行 配置 。 这 里 以 Windows 系统 为 例 ， 在 CA\WINDOWS 下 找到 php.ini 
文件 ， 定 位 到 如 下 位 置 : 

;extension=php_pdo.dll 

;extension=php_pdo_mysql.dll 

将 这 两 行 前 面 的 “;” (ini 文件 的 注释 符 ) 去 掉 ， 并 保存 php.ini 文件 。 然 后 定位 到 PHP 安装 目录 的 ext 
文件 夹 下 ， 在 该 文件 夹 中 查看 是 否 存 在 php_pdo_mysql.dll 和 php_pdo.dll 文件 ， 如 果 不 存在 ， 则 需 下 载 并 存 
入 ext 文件 夹 中 。 至 此 PDO_MYSQL 扩展 加 载 成 功 。 

至 此 ， 重 新 启动 Apache 服务 器 ，Zend Framework 运行 环境 配置 成 功 。 


外 wt 站 由 于 Zend Framework 完全 采用 PHPS 的 面向 对 和 象 编程 方式 实现 ， 所 以 PHP 版 本 必须 为 PHP 5 
以 上 。 同时， 为 了 发 挥 Zend Framework 各 个 组 件 的 功能 ， 建 议 使 用 PHP 5.2.6 及 以 上 版 本 。 如 果 是 使 用 
集成 化 安装 包 AppServ 来 搭建 PHP 开发 环境 ， 那 么 需要 下 载 AppServ 2.5.10 或 者 更 高 版 本 。 


25.2.2 ”框架 结构 
要 在 项 目 开 发 中 应 用 Zend Framework 框架 ， 只 完成 环境 的 配置 是 不 够 的 ， 还 要 搭建 一 个 完整 的 框架 结 
构 ， 才 能 在 项 目 开 发 中 发 挥 框架 的 作用 。 


自 Zend Framework 1.8 之 后 推出 了 Zend_Application 组 件 ， 使 搭建 结构 更 加 方便 和 系统 化 。 如 图 25.2 
所 示 为 一 个 相对 完整 的 Zend Framework 框架 结构 。 


25.2.3 ”创建 流程 


了 解 框架 的 结构 后 ， 下 面 讲 解 使 用 Zend Framework 建立 多 模块 MVC 框架 结构 的 具体 流程 。Zend 
Framework 程序 开发 的 基本 流程 如 图 25.3 所 示 。 


创建 引导 立 件 intex php 
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人 2 午 立 件 sprli cation ini 
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图 25.2 Zend Framework 框架 结构 图 25.3 ”最 基本 的 Zend Framework 程序 开发 步骤 
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例 25.1 开发 一 个 简单 的 实例 ， 输 出 “Hello，Zend Framework”， 熟 悉 Zend Framework 框架 的 基本 流 
程 。 (实例 位 置 : 光盘 \TMNInstance\2S\2S.1) 
(1) 在 public 目录 下 建立 URL 重 写 文件 .htaccess， 并 在 该 文件 中 输入 URL 重 写 规则 ， 如 下 所 示 : 
# 开 启 URL 重 写 
RewriteEngine on 
# 除 扩展 名 为 js、.css、.gif、.jpg、.png、.bmp 的 文件 外 ， 访 问 其 他 文件 都 转向 到 index.php 引导 文件 
RewriteRule \.(jslcsslgifljpglpnglbmp)$ index.php 


区 六 由 于 htaccess 文件 没有 文件 主 名 ， 所 以 在 Windows 系统 下 无 法 直接 命名 ， 下 面 介绍 两 种 创 
建 .htaccess 文件 的 方法 。 

方法 一 : 通过 Windows 系统 的 copy con 命令 创建 ， 如 图 25.4 所 示 。 创 建 完 成 后 按 CtrltZ 组 合 键 退 
出 编辑 模式 。 


画 Eea cowneoreomerazendee | 


图 25.4 通过 命令 提示 符 创 建 .htaccess 文件 
方法 二 : 通过 “记事 本 ”文本 编辑 工具 将 文件 另存 为 文件 名 为 .htaccess 的 文件 ， 这 里 应 该 注意 ， 保 
存 文 件 时 需要 选择 文件 类 型 为 所 有 文件 。 


(2) 在 public 目录 下 创建 ndex.php 引导 文件 。 代 码 如 下 : 
defined(APPLICATION_PATH') || define(APPLICATION_PATH' realpath(dimame(_FILE_) . /./application);: 
Ul 路 


defined('APPLICATION_ENV') || define(APPLICATION_ENV， getenv(APPLICATION_ENV) ? getenv 


(APPLICATION_ENV') : 'project'); /应 用 环境 
$arraylncludePath = array(".", realpath(dirmname(__FILE_) . '/../../../library'")); /指定 工程 包含 目录 
set_include_path(implode(PATH_SEPARATOR, $arraylncludePath)); // 将 指定 路 径 包 含 到 工程 中 
require_once 'Zend/Application.php'; // 包 含 Application.php 文 件 


$application = new Zend_Application(APPLICATION_ENV, APPLICATION_PATH . /configs/application.ini); 
// 实 例 化 Zend_Application 类 


$application->bootstrap()->run(); /调用 启动 文件 并 运行 项 目 

上 述 代码 中 ， 首 先 定 义 两 个 常量 APPLICATION_PATH 和 APPLICATION_ENV， 分 别 用 来 保存 应 用 路 
径 和 应 用 环境 名 ， 这 样 在 工程 的 其 他 模块 中 就 可 以 直接 使 用 ， 然 后 将 当前 目录 地 址 “.” 和 Zend Framework 
类 库 地 址 作为 数组 元 素 保存 在 名 为 $arrayIncludePath 的 数组 中 ， 并 使 用 implode0 函 数 将 上 述 路 径 用 
PATH_SEPARATOR 常量 连接 , 最 后 使 用 set_include_path0 方 法 将 路 径 导 入 到 工程 中 , 这 样 工程 就 可 以 直接 
找到 这 些 路 径 下 的 文件 。 

Zend Framework 类 库 被 导入 到 工程 后 ， 就 可 以 使 用 require_once 等 文件 包含 语句 包含 Zend 目录 下 的 
Application.php 文件 ， 之 后 实例 化 Zend_Application 类 ， 并 通过 引用 该 类 的 bootstrap0 方 法 调用 启动 类 ， 最 
终 通过 调用 run0 方 法 运行 工程 。 


人 5 注 总 在 文件 的 结尾 部 分 并 没有 加 入 “?>”， 因 为 在 文件 的 结尾 它 并 不 是 必需 的 。 这 样 就 可 以 避免 产 
生 一 些 难于 调控 的 错误 问题 。 例 如 ， 在 使 用 header0 函 数 来 重新 定向 (redirect) 时 ， 若 在 其 前 面 某 个 包 
含 文件 中 “?>” 的 后 面 加 上 空格 就 会 出 现 错误 。 

@ 
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(3) 在 第 (2) 步 实 例 化 Zend_Application 类 时 ， 为 类 的 构造 函数 传 入 一 个 名 为 application ini 的 文件 ， 

该 文件 是 工程 的 配置 文件 。 下 面 介绍 application.ini 文件 的 创建 过 程 。 其 基本 内 容 如 下 : 

[project] 

# 设 置 错误 级 别 

phpSettings.display_startup_errors = 1 

phpSettings.display_errors = 1 

# 设 置 时 区 

phpSettings.date.timezone = 'Asia/Shanghar 

# 配 置 启 动 类 

bootstrap.path = APPLICATION_PATH "/Bootstrap.php" 

bootstrap.class = "Bootstrap" 

# 配 置 工程 模块 

resources.frontController.moduleDirectory = APPLICATION_PATH "/modules" 


配置 文件 存放 位 置 和 名 称 并 不 固定 ， 只 要 与 实例 Zend_Application 类 传递 的 参数 一 致 即 可 。 


ot 这 里 配置 的 是 多 个 工程 模块 ，moduleDirectory=APPLICATION PATH "/modules"， 默 认 控 制 器 
放置 在 default 文件 夹 下 ， 文 件 夹 架构 如 图 25.5 所 示 。 如 果 配 置 的 是 单个 工程 模块 ，controllerDirectory= 
APPLICATION PATH "/controllers"， 文 件 夹 架 构 如 图 25.2 所 示 。 


(4) 在 application.ini 配置 文件 中 指定 启动 类 的 位 置 后 ， 工 程 就 可 以 找到 启动 文件 Bootstrap.php。 其 关 
键 代码 如 下 : 
class Bootstrap extends Zend_Application_Bootstrap_Bootstrap{ 
// 基 本 的 启动 类 可 以 不 进行 任何 操作 


让 
从 上 述 代 码 可 知 ， 启 动 类 Bootstrap 继承 自 类 Zend_Application_ Bootstrap_Bootstrap， 该 类 中 已 经 实现 最 
基本 的 启动 配置 工作 ， 所 以 在 启动 类 中 不 需 包含 任何 代码 就 可 以 运行 最 基本 的 Zend Framework 应 用 。 
(5) 完成 以 上 步骤 后 ， 就 可 以 创建 默认 控制 器 IndexController， 同 样 一 个 最 基本 的 控制 器 应 该 包含 一 
个 首页 动作 indexAction。 其 代码 如 下 : 
class IndexController extends Zend_Controller_Action{ 
public function indexAction (X{ /默认 动作 
Sthis->view->testStr="Hello Zend Framework!"; // 为 视图 变量 赋值 
} 
出 


p94 
说 明 使 用 Zend Framework 时 ， 控 制 器 名 应 该 使 用 Controller 结束 ， 动 作 名 应 以 Action 结束 。 视 图 
文件 夹 和 控制 器 文件 夹 必 须 在 同一 目录 下 ， 这 里 都 存储 在 application\modules\default 文件 夹 下 。 


(6) 完成 默认 控制 器 及 首页 动作 的 创建 后 ， 就 可 以 创建 视图 文件 index.phtml， 为 了 查看 效果 ， 在 视图 
文件 中 仅 输出 首页 动作 中 指定 的 视图 变量 的 值 。 其 代码 如 下 : 
echo $this->testStr; 


至 此 ， 多 模块 MVC 框架 结构 创建 完成 ， 其 完整 的 文件 夹 架 构 如 图 25.5 所 示 。 

假设 上 述 工 程 所 在 目录 为 Apache 默认 主 目录 的 TM\25W5.1 子 目 录 下 , 则 可 以 通过 如 下 URL 进行 访问 。 
http://127.0.0.1:82/TM/25/25.1/public/。 

http://localhost:82/TM/25/25.1/public/index。 

http://localhost:82/TM/25/25.1/public/index/index。 

其 运行 效果 如 图 25.6 所 示 。 


局 


第 25 章 Zend Framework 框架 


着 alc Zend Frarewerl ntemer Explorer 


ED hueyfocalhosts2/ T2979/pbiot |43|X Pes DP~ 
En cE ZEEEETTT 
安 Wias 下 -看 Heloz- <» ER 
RE Te 
引导 文件 3 CE TE] 本 不 100% ~ 
图 25.5 多 模块 MVC 框架 结构 图 25.6 Zend Framework 框架 创建 演示 


正常 情况 下 ，Zend Framework 的 访问 URL 应 该 为 “域名 + 模块 名 + 控制 器 名 + 动作 名 ”， 那 么 为 什么 上 
述 URL 省 略 了 部 分 内 容 也 可 以 访问 呢 ? 这 是 因为 在 Zend Framework 中 , 默认 模块 名 、 默认 控 制 器 名 和 首页 
动作 名 可 以 省 略 。 


25.2.4 ”Zend Framework 的 编码 标准 


程序 人 员 必 须 养 成 良好 的 编码 习惯 ， 为 编码 打下 一 个 良好 的 基础 。 在 每 一 个 项 目 中 都 会 存在 一 种 编码 
标准 ， 这 种 编码 标准 会 提高 程序 质量 ， 减 少 BUG， 使 维护 和 二 次 开发 变 得 轻松 。Zend Framework 编码 标准 
是 针对 Zend Framework 框架 开发 所 设计 的 ， 它 可 以 为 参与 Zend Framework 开发 的 个 人 和 团队 所 使 用 ,也 可 
以 在 没有 特定 编码 标准 的 项 目 中 使 用 。Zend Framework 编码 标准 如 下 。 

1. 命名 约定 

回 ”类 : 类 的 名 称 只 允许 使 用 字母 和 数字 ， 这 里 不 鼓励 使 用 数字 ， 如 果 类 名 称 中 包含 多 个 单词 ， 则 每 
个 单词 的 第 一 个 字母 必须 大 写 。 其 中 下 划 线 只 被 允许 分 隔 路 径 。 例 如 ，Zend/Application.php 文件 
中 对 应 的 类 名 称 应 该 为 Zend_Application。 

加 文件 名 : 文件 名 称 只 允许 使 用 字母 、 数 字 、 下 划 线 和 短 横 线 (-) ， 不 可 使 用 空格 。 同 时 ， 任 何 存 
在 PHP 代码 的 文件 都 必须 以 .php 扩展 名 结尾 〈 视 图 文件 除外 ) 。 

回 ”函数 和 方法 : 函数 名 称 只 允许 使 用 字母 和 数字 ， 不 允许 使 用 下 划 线 。 这 里 同样 不 鼓励 使 用 数字 。 
函数 的 名 称 总 是 以 小 写 开 始 ， 当 包含 多 个 单词 时 ， 每 个 单词 的 第 一 个 字母 大 写 。 在 对 象 方法 中 被 
定义 为 private 或 者 protected 的 ， 名 称 必 须 以 一 个 下 划 线 开始 。 

回 ”变量 : 变量 名 称 只 允许 使 用 字母 和 数字 ， 不 允许 使 用 下 划 线 。 这 里 同样 不 鼓励 使 用 数字 。 变 量 的 
名 称 总 是 以 小 写 开始 ， 当 包含 多 个 单词 时 ， 每 个 单词 的 第 一 个 字母 大 写 。 在 对 象 方法 中 被 定义 为 
Private 或 者 protected 的 ， 名 称 必 须 以 一 个 下 划 线 开始 。 

常量 : 常量 名 称 只 允许 使 用 字母 、 数 字 和 下 划 线 。 常 量 名 称 所 有 字母 都 必须 为 大 写 。 每 个 单词 都 
必须 以 下 划 线 分 隔 。 在 对 象 方法 中 定义 常量 必须 通过 const 定义 类 的 成 员 ， 这 里 不 鼓励 使 用 define 
定义 常量 。 


ot 除了 类 名 称 之 外 ， 其 余 命 名 都 鼓励 使 用 长 字符 ， 并 尽量 采用 能 够 说 明 意图 和 行为 的 单词 。 

编码 风格 

缩 进 : 缩 进 由 4 个 空格 组 成 。 需 要 注意 的 是 ， 不 允许 使 用 Tab 键 。 

行 长 度 : 一 行 代码 的 最 佳 长 度 在 80 个 字符 以 内 。 如 果 情 况 特殊 ， 最 多 可 以 延长 到 120 个 字符 。 
数组 : 当 使 用 数组 时 ， 每 个 逗号 后 都 用 一 个 空格 来 提高 可 读 性 。 如 果 是 关联 数组 ， 则 把 代码 分 为 


四 四 四 六 
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多 行 ， 在 每 一 个 连续 的 开头 用 空格 补充 并 对 齐 键 和 值 。 例 如 : 
$Array = array( 'oneKey => 'oneValue', 
'twoKey' => 'twoValue’); 
3. 注释 文档 
无 论 是 在 类 中 , 还 是 在 包含 PHP 代码 的 文件 中 , 必须 至 少 在 文件 顶部 包含 一 些 注释 信息 , 对 类 或 者 PHP 
代码 进行 解释 说 明 。 其 代码 如 下 : 
Jp 


* 文件 /类 的 简短 描述 
* 文件 /类 的 详细 描述 〈 如 果 有 的 话 ) … … 
到 


函数 和 方法 信息 则 必须 在 封装 的 函数 上 方 给 出 注释 信息 。 代 码 如 下 : 
J 


* 所 有 可 能 的 返回 值 〈 类 型 ) 

* @return void 

* 如 果 函 数 /方法 抛 出 异常 ， 使 用 @throws 于 所 有 已 经 知道 的 异常 类 中 
* @throws Zend_Application_Exception 


25.3 Zend Auth 身份 认证 


Zend_ Auth 是 登录 模块 中 对 身份 进行 验证 的 组 件 ， 但 是 并 不 是 授权 。Zend_Auth 具体 作用 是 定义 一 些 证 
书 来 确定 身份 是 否 是 Auth 声明 的 。 其 中 这 些 “ 证 书 ” 被 叫做 Zend_Auth 适配器 。 


25.3.1 Zend_Auth 适配器 


Zend_Auth 适配器 被 用 来 认证 特定 的 服务 器 。 例 如 接受 认证 证 书 , 认证 后 返回 的 结果 对 于 Zend_Auth 适 
配器 都 是 通用 的 。 身 份 认 证 必须 符合 程序 要 求 ， 这 就 需要 编写 适合 程序 的 适配器 类 。 

Zend_Anuth 适配器 类 必须 实现 Zend_Auth_Adapter_Interface 类 ， 调 用 接口 函数 authenticate0， 执 行 认证 
查询 。 例 如 ， 简 单 介绍 一 个 身份 认证 适配器 类 ， 仅 仅 针对 固定 用 户 名 和 密码 。 程 序 代 码 如 下 : 

<?php 

ps 


* 认证 用 户 名 和 密码 的 适配器 
J 


class Model_AuthAdapter implements Zend_Auth_Adapter_Interface{ 
protected $_username; 
protected $_password; 
te 
* Construct 
六 
* 带 入 用 户 名 和 密码 
* @return void 
i/ 


public function __construct($username,$password)}{ 


S$this->_username = $username; 
@ 
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Sthis->_password = $password; 
} 
je 
* 对 用 户 名 和 密码 进行 身份 认证 


* @throws Zend_Auth_Adapter_Exception If authentication cannot 
be performed 

* @return Zend_Auth_Result 

fh 


public function authenticate(X{ 
$array = array(); 
if (($this->_username == 'mr) && (($this->_password == 'mrsoft)))f 


$array[0] = true; 
@ return new Zend_Auth_Result(1,$array); 
jelse{ 
$array[0] = false; 
return new Zend_Auth_Result(-1,$array); 
有 
上 
} 
7 
| 目 技 万 


@ Zend Auth Result 方法 : 将 参数 带 入 Zend Auth Result 对 象 ， 验 证 时 只 需 调 用 
Zend_Auth Result 对 象 中 的 isValid0 方 法 。 

Zend Auth 适配器 返回 一 个 带 有 authenticate0) 的 Zend Auth Result 实例 。 在 Zend Auth Result 对 象 
中 ， 提 供 如 下 4 个 方法 对 Zend_Auth 适配器 返回 结果 进行 操作 。 

jsValid0: 返回 true 当 且 仅 当 结果 表示 一 个 成 功 的 认证 尝试 。 

getCode0: 返回 一 个 Zend_Auth Result。 常 量 标识 符 用 来 决定 认证 失败 的 类 型 或 者 是 否认 证 成 

功 。 可 以 用 于 开发 者 希望 区 别 若干 认证 结果 类 型 的 情形 。 
回 getIdentity0: 返回 认证 尝试 的 身份 。 
回 getMessages0: 返回 认证 尝试 失败 的 数组 。 


25.3.2 ”身份 持久 认证 


身份 的 持久 认证 必须 结合 Zend_Session 组 件 ，Zend_Session 组 件 通过 会 话 (session) 实现 服务 器 端 和 客 
户 端 之 间 的 持久 联系 。Zend_Session 采用 Zend_Session Namespace 对 象 的 方法 管理 , 而 且 会 话 空 间 (Session 
Namespaces) 提供 了 命名 空间 来 管理 会 话 数据 。 这 样 做 的 好 处 是 可 以 在 不 同 页 面 间 调用 不 同 的 Session 会 话 空间 。 
Zend_Session 主要 使 用 方法 如 下 : 
<?php 
include 'Zend/Session/Namespace.php’; 
$nameSpace = new Zend_Session_Namespace(Namespace'); /开启 Session 
下 面 通 过 实例 讲解 如 何 使 用 Zend_Auth 和 Zend_Session 对 身份 持久 认证 。 
例 25.2 结合 应 用 Zend_Auth 和 Zend_Session 来 对 身份 持久 认证 。( 实 例 位 置 :光盘 \TMNInstance\2S\25.2) 
(1) 将 Zend_ Auth 适配器 类 带 入 本 实例 中 ， 放 入 models 文件 夹 。 根 据 Zend Framework 编码 标准 重新 
编写 适配器 类 名 称 。 代 码 如 下 : 
class Model_AuthAdapter implements Zend_Auth_Adapter_Interface{ 
protected $_username; // 声 明成 员 变量 
protected $_password; // 声 明成 员 变量 


_ 国 
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public function construct($username,$password){ /创建 构造 函数 
Sthis->_username = $username; /为 成 员 变 量 赋值 
Sthis->_password = $password; // 为 成 员 变 量 赋值 

} 

public function authenticate(){ /定义 方法 对 身份 进行 认证 


$array = array(); 
if (($this->_username == 'mr') && (($this->_password == 'mrsoft))){  // 判 断 用 户 名 和 密码 
$array[0] = true; 


return new Zend_Auth_Result(1,$array); /正确 返回 结果 
jelse{ 
$array[0] = false; 
return new Zend_Auth_Result(-1,$array); /| 错误 返回 结果 
» 
» 
} 


p94 
< 说 明 在 Model 模型 层 写 入 的 类 可 以 直接 在 程序 中 实例 化 ， 而 不 需要 引用 文件 。 


(2) 控制 器 中 功能 分 为 验证 用 户 名 和 密码 、 成 功 页 面 输出 和 安全 退出 。3 个 功能 分 别 对 应 3 个 方法 。 
其 程序 代码 如 下 : 
<?php 
class IndexController extends Zend_Controller_Action{ 
public function indexAction(){ 
$this->view->assign("title"," 网 站 登录 界面 "); 


0 if($this->_request->isPost()\{ // 是 否 POST 传输 
$username = S$this->_request->getPost('username’); /取得 username 值 
$password = $this->_request->getPost(password'); /取得 password 值 
$auth = Zend_Auth::getlnstance(); // 调 用 Zend_Auth 


$authModel = new Model_AuthAdapter($username, $password); 
// 将 适配器 结果 放 入 Zend_Auth 的 结果 集中 


© Sresult = $auth->authenticate($authModel); 

if($result->isValid()}{ // 使 用 默认 验证 
/开启 SESSION 

$sessionNameSpace = new Zend_Session_Namespace('project'); 
$sessionNameSpace->username = $username; /赋予 session 值 
$sessionNameSpace->password = $password; 

Sthis->_redirect('index/success'); // 跳 转 到 成 功 页 面 
}else{ 

echo "用 户 名 和 密码 错误 "; 

. 

} 


public function successAction(){ 

$this->view->assign('title', ' 成 功 界 面 '); 

$sessionNameSpace = new Zend_Session_Namespace('project); 
// 判 断 session 中 是 否 设 定 usemame 变量 ， 即 是 否 登录 


© if($sessionNameSpace->username == ™")/{ 
die(" 不 允许 直接 访问 此 页 面 "); 
} 


S$this->view->username = $sessionNameSpace->username; 
S$this->view->password = $sessionNameSpace->password; 
} 

®@ publicfunction logoutAction(){ 


@ 
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$sessionNameSpace = new Zend_Session_Namespace('project'); 
unset($sessionNameSpace->username); /| 销毁 username 变量 
Sthis->_redirect('index/index'); 

} 

j 


SC 名 曙 
@ Sthis->_ request->isPost( 方 法 : 判断 当前 页 面 是 否 使 用 POST 方式 提交 。 
@ authenticate() 方 法 : 适配器 类 中 调用 Zend_ Auth Result 对 象 方法 。 
四 加 入 不 允许 未 登录 访问 本 页 面 的 功能 。 通 过 session 判断 。 
@ 安全 退出 方法 : 销毁 session 变量 。 


(3) 从 控制 器 中 可 以 看 出 存在 3 个 页 面 : index、success 和 logout。 但 是 仔细 推敲 后 发 现 真正 输出 数据 
的 页 面 只 有 index 和 success,logout 页 面 没有 任何 数据 输出 .所 以 视图 层 中 只 建立 index.phtml 和 success.phtml 
两 个 页 面 。index.phtml 页 面 关键 代码 如 下 : 
@<title><?php echo $this->escape($this->title); ?></title> 
<style type="text/css"> 
.Submit{ 
font-weight:bold; 
width:112px; 
bordermedium none; 
@ background:#FFF url(<?php echo $this->baseUrl("images/index_02.jpg"); ?>) no-repeat; 


cursor:pointer; 
line-height:33px; 
font-size: 14px; 

B 

</style> 


目 <form id="form1" name="form1" method="post"” action="<?php echo S$this->Url(array('controller'=>'index', 
'action'=>'index'")); ?>"> 
<table width="338" height="168" background="<?php echo $this->baseUrl("images/index_01.jpg"); ?>"> 
<tr> 
<td height="22"><input name="username'" type="text" size="24" maxlength="10" /></td> 
</tr> 
<tr> 
<td><input name="password" type="password" size="25" maxlength="10" /></td> 
</tr> 
</table> 
<table width="338" height="62" background="<?php echo S$this->baseUrl("images/index_04.jpg"); ?>"> 
<tr> 
<td><input type="submit" name="submit" class="submit" value=" "/></td> 
<td><img src="<?php echo S$this->baseUrl("images/index_03.jpg"); ?>" border="0"/></td> 
</tr> 
</table> 
</form> 


SS 人 
@ Sthis->escape($this->title): 输出 标题 。 
@ Sthis->baseUrl("images/index 02.jpg"): 设置 背景 颜色 。 
@ $this->Url(array('controller=>'index'action'=>'index)): 设置 表单 中 数据 提交 的 路 径 。 
在 form 表单 中 的 action 表示 提交 数据 到 哪 一 个 程序 。Zend Framework 框架 中 ， 如 果 提 交 的 数据 不 
是 当前 控制 器 ， 必 须 使 用 $this->Url0 方 法 确认 控制 器 。 


S49 


PHP+MySQL 开发 实战 


success.phtml 页 面 代 码 如 下 : 
<title><?php echo $this->escape($this->title); ?></title> 
<table width="300" border="1" cellpadding="0" cellspacing="0" bordercolor="#333333"> 
<tr height="30"> 
<td width="60" align="center"> 用 户 名 : </td> 
<td><?php echo $this->escape($this->username); ?></td> 
</tr> 
<tr height="30"> 
<td align="center'> 密 &nbsp; 码 : </td> 
<td><?php echo S$this->escape($this->password); ?></td> 
</tr> 
<tr height="30"> 
<td><a href="<?php echo $this->Url(array('controller=>'index',action'=>'\logout)); ?>"> 安 全 退出 </a></td> 
</tr> 
</table> 


| 


| 隐 二 在 视图 层 中 ， 如 果 要 调用 图 片 或 者 各 种 文件 ， 可 以 使 用 Zend Framework 中 的 baseUrlO 函 数 ， 
这 样 调用 的 位 置 就 是 入 口 文件 所 在 文件 类。 


其 运行 结果 如 图 25.7 所 示 。 


25.3.3 数据库 认证 


Zend Framework 框架 中 通过 Zend_Auth_Adapter_ DbTable0 组 件 实现 数据 库 认证 。 

首先 ， 应 用 Zend_Db_Adapter_ Abstract 连接 一 个 数据 库 ， 并 将 返回 对 象 传递 给 Zend_Auth Adapter_ 
DbTable 的 构造 器 。 

然后 , 实例 化 Zend_Auth_Adapter_DbTable， 并 通过 配置 选项 传递 参数 ， 以 及 用 户 登录 的 用 户 名 和 密码 ， 


实现 数据 库 的 认证 。 
可 用 的 配置 选项 如 下 。 
tableName: 包含 认证 证 书 的 数据 库 表 名 ， 执 行 数 据 库 认 证 查询 需要 依靠 该 证 书 。 
identityColumn: 数据 库 表 的 列 名 称 ， 用 来 表示 身份 。 身 份 列 必须 包含 唯一 的 值 ， 例 如 用 户 名 或 者 
E-mail 地 址 。 
credentialColumn: 数据 库 表 的 列 名 称 ， 用 来 表示 证 书 。 在 一 个 简单 的 身份 和 密码 认证 scheme 下 ， 
证 书 的 值 对 应 为 密码 。 


credentialTreatment: 在 许多 情况 下 ， 密 码 和 其 他 敏感 数据 是 加 密 的 。 通 过 指定 参数 化 的 字 串 来 使 
用 这 个 方法 ， 如 MD5(?) 或 者 PASSWORD(?)， 开 发 者 可 以 在 输入 证 书 数据 时 使 用 任意 的 SQL 


语句 。 
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最 后 ， 调 用 authenticate() 方 法 执行 认证 查询 。 在 认证 结果 对 象 中 ， 可 以 通过 getIdentity0 方 法 获取 认证 
的 身份 ， 也 可 以 通过 getResultRowObject0 方 法 返回 一 行 数据 。 


25.4 ”Zend Db 数据 库 操 作 


Web 语言 编程 离 不 开 数 据 库 的 支持 。Zend Framework 框架 中 连接 和 操作 数据 库 的 是 Zend_Db 组 件 。 
Zend_Db 组 件 是 一 个 基于 PDO 模块 的 数据 库 抽象 层 API, 它 可 以 支持 多 种 数据 库 , 如 MySQL、 SQLite、 SQL 


server 等 。 


25.4.1 Zend_Db_Adapter 数据 库 操作 


应 用 Zend_Db_Adapter 必须 静态 调用 Zend_Db::factory0 方 法 。 例 如 ， 调 用 PHP 的 最 佳 搭档 MySQL 数 
据 库 。 程 序 代 码 如 下 : 
<?php 


include_once 'Zend/Db.php'; 
$params = array( 


/包含 Zend_Db 文件 


"host => 'localhost', // 连 接 数据 库 URL 
username'=>root'， /连接 数 据 库 用 户 名 
'password'=>"111', // 连 接 数据 库 密码 
dbname' =>'db_database25', /| 连接 数据 库 名 称 

) 

$db = Zend_Db::factory(PDO_MYSQL', $params): /| 基于 PDO 连接 MySQL 数据 库 


这 就 是 一 个 简单 的 连接 和 操作 数据 库 的 方法 。 但 是 由 于 Zend Framework 的 版 本 升级 到 1.8 以 上 ， 入 口 
文件 中 并 不 需要 调用 此 方法 ， 所 以 此 方法 已 经 不 能 够 满足 现在 的 需求 ， 当 然 并 不 是 不 可 以 使 用 。 下 面 介绍 
的 Zend_Db_Table 是 Zend Framework 推荐 的 连接 和 操作 数据 库 的 组 件 。 


25.4.2 Zend_Db_Table 数据 库 操作 


Zend_Db_Table 是 Zend Framework 的 表 模 块 。 它 通过 zend_db_adapter 连接 到 数据 库 ， 创 建 类 来 继承 
Zend_Db_Table 类 ， 实 例 化 创建 的 类 ， 通 过 返回 的 对 象 对 数据 表 进 行 操作 和 查询 。 
例 25.3 在 Zend Framework 1.8 及 以 上 版 本 中 连接 和 操作 MySQL 数据 库 ， 查 询 表 tb_user 中 数据 ， 并 
输出 查询 结果 。〔 实 例 位 置 ， 光盘 \TMNInstance\25\25.3) 
(1) 搭建 Zend Framework 的 基本 环境 。 
(2) 使 用 MySQL 数据 库 ， 根 据 Zend Framework 1.8 以 上 版 本 资源 加 载 方法 〈 详 细 情 况 查 看 25.2.3 基 
本 环境 搭建 中 的 启动 文件 ) ， 将 连接 数据 库 信息 写 入 配置 文件 。 程 序 代码 如 下 : 


resources.db.adapter = "PDO_MYSQL" 
resources.db.params.host = "localhost" 
resources.db.params.username = "root" 
resources.db.params.password = "111" 


resources.db.params.dbname = "db_database25" 
resources.db.params.driver_options.1002 = "set names utf8" 


/| 基于 PDO_MYSQL 连接 数据 库 
/数据 库 URL 

/| 数据 库 用 户 名 

/数据库 密码 

/数据 库 名 称 

/| 数据 库 字符 集 


(3) 在 启动 文件 中 ， 通 过 _init 方法 加 载 数据 库 资源 。 程 序 代码 如 下 : 


protected function _initDB(){ 


@ $options = $this->getOption(resources'); 
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$options = $options['db"]; 

S$resources = $this->getPluginResource('db"); 

$db = $resources->getDbAdapter(); 
Zend_Db_Table::setDefaultAdapter($db); 
Zend_Registry::set('dbAdapter', $db); 
Zend_Registry::set('dbprefix', $options['params'][ prefix"]); 


SO 涪 明 @ getOption() 方 法 : 在 配置 文件 中 找到 resources 开头 的 资源 配置 。 
@ $options['db']: 获取 数据 库 资源 ， 在 资源 配置 中 找到 数据 库 (db ) 项 。 
@ getPluginResource() 方 法 : 获取 系统 中 数据 库 资源 。 
@ Sresources->getDbAdapter0): 定义 数据 库 操作 变量 。 在 程序 中 只 要 通过 调用 Zend Registry::get 方 
法 调用 此 变量 ， 即 可 实现 对 数据 库 的 操作 。 


(4) 通过 Zend_Registry::get 方法 调用 数据 库 资源 变量 ， 通 过 fetchAll 方法 执行 查询 语句 ， 最 后 通过 
print r 输出 查询 结果 。 程 序 代码 如 下 : 
public function indexAction(){ 
// 调 用 db 对 象 ，dbAdapter 为 启动 文件 中 设置 调用 资源 
Sthis->_db = Zend_Registry::get(dbAdapter); 


$sql = "select * from tb_user"; /SQL 语句 
$result = S$this->_db->fetchAll($sql); // 得 到 SQL 语句 结果 ， 查 看 fetchAll 
print_r($result); 


， 

运行 本 实例 ， 查 看 数据 表 中 信息 ， 会 抛 出 控制 器 错误 ， 如 图 25.8 所 示 。 

这 是 因为 Zend Framework 框架 找 不 到 控制 器 (IndexController) 对 应 的 视图 文件 ， 所 以 需要 在 视图 文件 
夹 (views) 下 创建 一 个 index.phtml 视图 页 面 ， 在 index.phtml 页 面 中 可 以 不 加 载 任何 程序 。 本 实例 的 运行 效 
果 如 图 25.9 所 示 。 


图 25.9 运行 效果 图 


25.4.3 ”数据 表 类 


通过 继承 Zend_Db_Table 类 ， 可 以 完成 对 数据 表 的 操作 ， 而 对 数据 表 的 操作 多 数 都 是 代码 的 重用 ， 所 
以 最 好 的 方法 就 是 将 其 定义 到 模型 中 ， 这 样 可 以 提高 代码 的 重用 率 ， 同 时 也 便于 对 程序 进行 更 新 和 维护 。 

通过 模型 类 继承 Zend_Db_Table 类 ， 封 装 对 一 个 数据 库 的 操作 方法 是 非常 简单 的 。 数 据 表 类 中 对 数据 
库 的 操作 主要 分 为 4 种 。 

insert($dataArray): 添加 数据 ， 参 数 $dataArray 是 对 应 数据 表 的 联合 数组 。 

Update($setArray,$where): 修改 数据 ， 参 数 $setArray 是 对 应 数据 表 的 联合 数组 ， 参 数 Swhere 是 修改 


@ 
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数据 的 条 件 ， 相 当 于 where 语句 。 
delete($where): 删除 数据 ， 参 数 Swhere 是 删除 数据 的 条 件 ， 相 当 于 where 语句 。 
fetchAll($where = null Sorder = null, $count = null, Soffset = null): 查询 数据 。 如 果 是 单纯 的 查询 操 
作 可 以 直接 使 用 fetchAll0 方 法 。 但 是 ， 当 查询 有 很 多 附加 条 件 时 ， 就 要 使 用 fetchAl10 方 法 的 参数 ， 
其 参数 说 明 如 下 。 
> Swhere: 表示 where 语句 。 
> ”Sorder: 表示 order by 语句 。 
> $count: 表示 limit 语句 中 显示 多 少 结果 行 。 
> ”Soffset: 表示 limit 语句 中 从 第 几 个 数据 开始 。 
例 25.4 创建 数据 表 类 ， 通 过 数据 表 类 完成 对 数据 库 的 添加 、 删 除 、 修 改 和 查询 。〔 实 例 位 置 ， 光 盘 \ 
TM™\Instance\25\25.4) 
(1) 在 public 目录 下 建立 URL 重 写 文件 .htaccess， 并 在 该 文件 中 输入 URL 重 写 规则 。 代 码 如 下 : 
# 开 启 URL 重 写 
RewriteEngine on 
# 除 扩展 名 为 js、.css、.gif、.jjpg、.png、.bmp 的 文件 外 ， 访 问 其 他 文件 都 转向 到 index.php 引导 文件 
RewriteRule N\.(jslcsslgiflipglpnglbmp)$ index.php 
(2) 在 public 目录 下 创建 index.php 引导 文件 。 代 码 如 下 : 
defined('APPLICATION_PATH') || define(APPLICATION_PATH', realpath(dirmame(_ FILE. ) . /../application’)): 
/应 用 路 径 


defined('APPLICATION_ENV') || define(APPLICATION_ENV， getenv(APPLICATION_ENV) ? getenv 


(APPLICATION_ENV') : 'project'); // 应 用 环境 
$arraylncludePath = array(… realpath(dirname(__FILE_) . /././library)); /指定 工程 包含 目录 
set_include_path(implode(PATH_SEPARATOR, $arraylncludePath)); /将 指定 路 径 包 含 到 工程 中 
require_once 'Zend/Application.php'; // 包 含 Application.php 文件 


$application = new Zend_Application(APPLICATION_ENV, APPLICATION_PATH . ‘/configs/application.ini"); 
/实例 Zend_Application 类 

$application->bootstrap()->run(); /调用 启动 文件 并 运行 项 目 

首先 , 定义 常量 APPLICATION PATH 和 APPLICATION_ENV, 分 别 用 来 保存 应 用 路 径 和 应 用 环境 名 。 
然后 ， 将 当前 目录 地 址 “.” 和 Zend Framework 类 库 地 址 作为 数组 元 素 保存 在 名 为 $arrayIncludePath 的 数组 
中 ， 并 使 用 implode0 函 数 将 上 述 路 径 用 PATH_SEPARATOR 常量 连接 。 最 后 ， 使 用 set_include_path0 方 法 
将 路 径 导 入 到 工程 中 ， 这 样 工 程 就 可 以 直接 找到 这 些 路 径 下 的 文件 。 

Zend Framework 类 库 被 导入 到 工程 后 ， 使 用 require_once 包含 语句 包含 Zend 目录 下 的 Application.php 
文件 ， 之 后 实例 化 Zend_Application 类 ， 并 通过 引用 该 类 的 bootstrap() 方 法 调用 启动 类 ， 再 调用 run0 方 法 运 
行 工程 。 

(3) 在 第 (2) 步 实例 化 Zend_Application 类 时 ， 为 类 的 构造 函数 传 入 一 个 名 为 application.ini 的 文件 ， 
即 工程 的 配置 文件 。 其 代码 如 下 : 

[project] 

# 设 置 错误 级 别 

phpSettings.display_startup_errors = 1 

phpSettings.display_errors = 1 

# 设 置 时 区 

phpSettings.date.timezone = 'Asia/Shanghai' 

# 配 置 启动 类 

bootstrap.path = APPLICATION_PATH "/Bootstrap.php" 

bootstrap.class = "Bootstrap" 

# 配 置 工程 模块 

resources.frontController.controllerDirectory = APPLICATION_PATH "/controllers" 

# 配 置 数据 库 PDO 模块 
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resources.db.adapter = "PDO_MYSQL" 
# 设 置 数据 库 服务 器 
resources.db.params.host = "localhost" 
# 设 置 服务 器 用 户 名 
resources.db.params.username = "root" 
# 设 置 服务 器 密码 
resources.db.params.password = "111" 
# 设 置 数据 库 
resources.db.params.dbname = "db_database25" 
# 设 置 数据 库 编码 格式 
resources.db.params.driver_options.1002 = "set names utf8" 

(4) 在 application.ini 配置 文件 中 指定 启动 类 的 位 置 后 ， 工 程 就 可 以 找到 启动 文件 Bootstrap.php。 其 关 

键 代码 如 下 : 
class Bootstrap extends Zend_Application_Bootstrap_Bootstrap{ 
public function _initAutoload(X{ 
$moduleAutoloader = new Zend_Application_Module_Autoloader(aray(namespace' => "basePath => "../application')); 
return $moduleAutoloader; 
} 
启动 类 Bootstrap 继承 自 类 Zend_Application_Bootstrap_Bootstrap， 该 类 中 已 经 实现 最 基本 的 启动 配置 
工作 。 
(5) 创建 模型 models， 创 建 数据 表 类 User， 定 义 操作 数据 表 的 方法 。 其 代码 如 下 : 


<?php 
class Model_User extends Zend_Db_Table{ /定义 数据 表 类 ， 继 承 Zend_Db_Table 
protected $_name = "tb_user"; // 定 义 数据 表 名 称 变量 
protected $_primary = "id"; /定义 数据 id 变量 
protected $_adapter; /定义 数据 库 连接 标识 变量 
public function init()f /定义 init() 方 法 
Sthis->_adapter = $this->getAdapter(); 1/ 获取 操作 数据 表 的 对 象 
上 


上 
(6) 创建 控制 器 controllers、 默 认 控 制 器 IndexController， 定 义 添 加 、 修 改 、 删 除 和 查询 动作 的 方法 ， 
实例 化 数据 表 (tb_user) 模型 中 定义 的 User 类 ， 通 过 返回 的 对 象 调用 数据 库 操作 方法 ， 完 成 对 数据 库 的 操 
作 。 其 关键 代码 如 下 : 
<?php 
class IndexController extends Zend_Controller_Action{ 
public function indexAction(){ 
Stable = new Model_User(); // 类 的 实例 化 
/增加 的 数据 
$data = array( 
metname' => 'King ', 
"password' => 'mrsoft,, 


六 
@ if($table->insert($data)\{ /执行 添加 操作 


Sthis->view->insert = "插入 数据 成 功 !"; 
) 
public function deleteAction(){ /定义 删除 方法 
Stable = new Model_User(); // 类 的 实例 化 
$where = id=25 /定义 删除 的 条 件 
@ if($table->delete($where)X{ /执行 删除 操作 


Sthis->view->delete = "删除 成 功 ! 


@ 
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Jelse { 
Sthis->view->delete = "该 数据 已 经 不 存在 "; 
; } 
public function updateAction(){ /定义 更 新 方法 
$table = new Model_User(); 1/ 类 的 实例 化 
$set = array( 
'netname’ => 'mingri', 
E 
$where = "id =2'; /定义 更 新 条 件 
目 if($table->update($set, $where)X{ /| 执行 更 新 操作 
S$this->view->update = "修改 成 功 !"; 
jelse{ 
$this->view->update = "该 数据 不 存在 或 已 经 被 修改 过 "; 
public function fetchAction(X{ /定义 查询 方法 
$table = new Model_User(); /| 类 的 实例 化 
// 定 义 查 询 条 件 
$where = null; 
S$order = "id desc"'; /以 上 D 降序 排列 
$count = 3; // 输 出 3 条 记录 
$offset = 0; // 从 第 1 条 记录 开始 
[3 Sresult_all = $table->fetchAll($where, $order, $count, $offset); /执行 查询 语句 


Sthis->view->select =$result_all; 


有 


p94 
ww 说 明 @ insert() 方 法 : 相当 于 执行 insert into tb_user(login_name,login_pwd) values(king', 'mrso 人 t) 语 句 。 
@ delete() 方 法 : 相当 于 执行 delete from tb_user where id=2 语句 。 
@ update() 方 法 : 相当 于 执行 update tb_user set login name = 'mingri' where id=2 语 和 句 。 
@ fetchAll0: 可 以 无 参数 直接 使 用 ， 但 是 当 使 用 参数 时 ， 参 数位 置 必 须 对 应 。 


(7) 控制 器 和 动作 方法 创建 完成 后 ， 创 建 与 控制 器 对 应 的 视图 文件 ， 这 里 创建 4 个 视图 文件 : 
index.phtml、update.phtml、delete.phtml 和 fetch.phtml， 输 出 对 数据 库 的 操作 结果 。 在 fetch.phtml 视图 文件 
中 ， 通 过 foreach 语句 循环 输出 数据 库 的 查询 结果 。 其 代码 如 下 : 

<html xmlns="http://www.w3.org/1999/xhtml"> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 

<title> 查 看 数据 </title> 

<link href="<?php echo S$this->baseUrl('css/style.css');?>" rel="stylesheet" type="text/css" /> 

</head> 

<body> 

<table width="600" border="1" cellpadding="1" cellspacing="1" bordercolor="#FFFFFF" bgcolor="#336600"> 
<tr> 


<td height="25" align="center" bgcolor="#FFFFFF">ID</td 
<td align="center" bgcolor="#FFFFFF"> 用 户 名 </td> 
<td align="center" bgcolor="#FFFFFF"> 密 码 </td> 

</tr> 

<?php foreach ($this->select as $key => $select){?> 

<tr> 
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<td height="20" align="center" bgcolor="#FFFFFF"><?php echo $select[id];?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo $select[login_name'];?></td> 
<td align="center" bgcolor="#FFFFFF"><?php echo $selectflogin_pwd'];?></td> 


</ltr> 
<?php }?> 
<tr> 


<td height="30" colspan="3" align="center" bgcolor="#FFFFFF"> 
<table width="300" border="0" cellpadding="0" cellspacing="0"> 
<tr> 
<td><a href="index"> 添 加 数据 </a></td> 
<td><a href="update"> 更 新 </la></td> 
<td><a href="delete"> 删 除 </a></td> 
<td height="28"><a href="fetch"> 查 看 </a></td> 


</tr> 
</table> 
<ltd> 
</tr> 
</table> 
</body> 
</html> 
运行 结果 如 图 25.10 所 示 。 
吕 用 各 I EE 
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图 25.10 查看 数据 库 中 数据 


25.5 Zend File 文件 控制 


Zend_File 组 件 主要 用 于 实现 文件 的 上 传 和 下 载 ， 该 组 件 设计 上 采用 适配器 方式 ， 这 样 无 论 使 用 HTTP 
的 POST 方式 还 是 使 用 FTP 方式 对 文件 进行 上 传 ， 其 具体 实现 代码 都 相同 ， 如 果 上 传 方式 发 生 改变 ， 只 需 
更 改 适 配器 即 可 。 

由 于 该 组 件 尚 不 成 熟 ， 目 前 只 支持 HTTP 的 POST 方式 上 传 ， 所 以 只 能 使 用 Zend_File_ Transfer 
Adapter_Http 适配器 实现 文件 的 上 传 ， 但 Zend Framework 开发 团队 不 久 将 推出 更 多 的 适配器 (如 支持 FTP 
方式 的 适配器 ) 。 本 节 将 主要 介绍 Zend_File_Transfer_ Adapter_Http 的 使 用 方法 。 


25.5.1 使 用 Zend_File_Transfer_Adapter_Http 实现 POST 方式 文件 上 传 


使 用 Zend_File_Transfer Adapter_Http 实现 文件 上 传 主要 使 用 该 类 的 setDestination0 和 receive0 方 法 实 
现 ， 其 中 setDestination0 方 法 用 于 指定 上 传 文件 保存 在 服务 器 端的 目录 ，receive0 方 法 用 于 执行 上 传 操作 。 
下 面 将 通过 一 个 简单 的 实例 讲解 使 用 该 适配器 类 实现 POST 方式 文件 上 传 的 方法 。 
例 25.5 使 用 Zend File Transfer Adapter Http 适配器 类 实现 文件 上 传 .( 实 例 位 置 : 光盘 \TM\Instance\ 
25W25.5) 
为 了 能 够 掌握 使 用 上 述 适 配器 类 实现 文件 上 传 的 最 核心 方法 ， 实 现 本 实例 时 只 建立 一 个 文件 上 传 表单 
和 一 个 用 于 实现 上 传 业务 逻辑 的 控制 器 类 。 


@ 
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(1) 建立 Zend Framework 的 MVC 框架 结构 ， 具 体 创建 过 程 请 参见 本 章 第 25.2 节 的 内 容 ， 这 里 不 再 袭 述 。 
(2) 建立 用 于 实现 文件 上 传 的 表单 。 为 了 便于 查看 上 传 效 果 ， 在 该 表单 中 只 设置 用 于 选择 上 传 文件 的 
文件 域 和 用 于 提交 表单 的 图 片 按钮 。 其 代码 如 下 : 
<form enctype="multipart/form-data" action="<?php echo $this->Url(array('controller=>'index',action'=>'index)); ?>" 
method="POST"> 
<span class="STYLE1"> 选 择 上 传 文件 :</span> 
<input name="uploadedfile" type="file" />&nbsp;&nbsp; 
<input type="image" name="imageField" src="<?php echo S$this->baseUrl(images/sc.jpg');?>" /> 
</form> 
通过 POST 方法 实现 文件 上 传 的 表单 form 标签 中 必须 设置 enctype 属性 的 值 为 multipart/form-data， 并 
且 method 属性 的 值 必须 为 POST。 
(3) 建立 上 传 表单 后 ， 就 可 以 编写 用 于 实现 文件 的 业务 逻辑 。 本 实例 实现 文件 上 传 的 代码 是 在 
IndexController 控制 器 类 的 indexAction 动作 中 。 其 关键 代码 如 下 : 
<?php 
class IndexController extends Zend_Controller_Action{ 
public function indexAction(}{ 


if($this->_request->isPost()X{ /l 潮 断 是 否 是 POST 上 传 
$adapter = new Zend_File_Transfer_Adapter_Http(); 
$adapter->setDestination('./upfiles'); // 设 置 上 传 文件 存储 路 径 
if(!$adapter->receive()\{ /| 执行 上 传 操作 

$messages = $adapter->getMessages(); /获取 返回 的 错误 信息 
echo implode("<br>",$messages); // 输 出 错误 信息 
jelsef 


echo "<script>alert(' 上 传 成 功 ! ');</script>"; 
} 
} 
} 


} 

在 上 述 代 码 中 , 首先 使 用 当前 控制 器 对 象 的 _request 属性 的 isPost0 方 法 判断 是 否 已 经 提交 了 表单 ， 如果 
是 则 进行 文件 上 传 操作 。 

在 具体 实现 文件 上 传 时 ， 首 先 通过 new 关键 字 实例 化 Zend_File_Adapter_Http0 适 配器 类 ， 然 后 调用 该 
适配器 类 的 setDestination() 方 法 指定 上 传 文件 要 保存 的 服务 器 端 目录 ， 最 后 通过 调用 该 适配器 类 的 receive() 
方法 实现 文件 上 传 。 这 里 需要 注意 ， 如 果 上 传 失败 ， 则 调用 receive0 方 法 后 将 返回 false 值 ， 并 可 以 通过 上 
述 适 配器 类 的 getMessages0 方 法 返回 上 传 失败 的 原因 。 

运行 上 述 实 例 ， 如 图 25.11 所 示 ， 在 表单 的 文件 域 中 选择 要 上 传 的 文件 ， 然 后 单 击 “ 上 传 ”按钮 ， 则 所 
上 传 的 文件 将 被 保存 到 该 实例 主 目录 的 upfiles 子 目 录 中 。 
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图 25.11 Zend File Transfer Adapter Http 实现 POST 方式 文件 上 传 
根据 上 述 实例 可 知 ， 通 过 Zend_File_Transfer Adapter_Http 适配器 类 实现 文件 上 传 的 关键 代码 只 有 如 下 


3 行 : 
$adapter = new Zend_File_Transfer_Adapter_Http(); /实例 适配器 类 
$adapter->setDestination('./upfiles'); /定义 上 传 文件 保存 路 径 
$adapter->receive(); /执行 上 传 操作 


@ 
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25.5.2 ”对 上 传 文件 的 合理 性 验证 


上 传 文件 时 ， 有 时 需要 对 文件 的 类 型 、 大 小 进行 限制 。 例 如 使 用 PHP 开发 的 Web 应 用 在 制作 文件 上 传 
模块 时 ， 为 了 提高 项 目的 安全 性 ， 不 允许 上 传 扩展 名 为 .php 或 .exe 的 文件 ， 那 么 使 用 Zend File_Transfer_ 
Adapter_Http 适配器 类 如 何 实现 上 传 文件 的 有 效 性 验证 呢 ? 

在 具体 讲解 使 用 Zend_File Transfer Adapter Http 适配器 类 实现 文件 上 传 的 合理 性 验证 前 ， 首 先 应 该 知 
道 常用 的 验证 规则 有 哪些 ， 其 中 表 25.2 所 示 为 该 适配器 类 常用 的 效 验 规则 。 


表 25.2 Zend_File_Transfer_Adapter_Http 适配器 类 常用 效 验 规则 


效 验 规则 说 明 
Count 指定 上 传 文件 的 数量 范围 验证 规则 
ExcludeExtension 指定 不 允许 的 上 传 文件 扩展 名 验证 规则 
Extension 指定 允许 的 上 传 文件 扩展 名 验证 规则 
ImageSize 指定 上 传 图 片 大 小 的 验证 规则 
IsImage 指定 上 传 文件 是 否 为 图 片 的 验证 规则 
size 指定 上 传 文件 大 小 的 验证 规则 
ExcludeMimeType 指定 上 传 文件 不 允许 的 mime 类 型 验证 规则 
MimeType 指定 上 传 文件 允许 的 mime 类 型 验证 规则 
Exists 指定 要 上 传 的 文件 已 经 存在 于 服务 器 的 验证 规则 
NotExists 指定 要 上 传 的 文件 不 存在 于 服务 器 的 验证 规则 


了 解 表 25.2 中 的 常用 验证 规则 后 ， 又 如 何 将 这 些 规 则 应 用 于 上 传 过 程 中 呢 ? 这 里 只 需 使 用 上 传 适配器 
类 的 addValidator() 方 法 在 上 传 过 程 中 指定 上 传 规则 即 可 ， 该 方法 一 般 包含 3 个 参数 ， 第 1 个 参数 用 于 指定 


表 25.2 中 的 验证 规则 名 称 


; 第 2 个 参数 为 boolean 型 ， 如 果 为 tue 则 当 该 条 验证 失败 时 继续 向 下 验证 ， 否 则 


不 再 继续 向 下 验证 ， 第 3 个 参数 为 数组 型 ， 用 于 设置 具体 验证 方式 和 验证 错误 的 提示 信息 。 
例如 ， 在 图 片上 传 时 使 用 Zend_File_Transfer_ Adapter_Http 适配器 类 验证 图 片 大 小 只 能 小 于 1MB 的 代 


码 如 下 。 
$adapter = new Zend_File_Transfer_Adapter_Http(); /实例 化 适配器 类 
$adapter->setDestination('./upfiles'); /定义 上 传 文件 保存 路 径 
$adapter->addValidator('Size', false, array('min' => '0kB' , 'max' => '1MB' , 'bytestring' => false , ,messages' => 
” 您 所 上 传 的 图 片 不 能 超过 1MB')); // 对 文件 大 小 进行 验证 


$adapter->receive(); 


/执行 上 传 操 作 


25.5.3 ”为 上 传 增加 过 滤 规 则 


通过 适配器 类 的 addValidator0 方 法 可 以 对 上 传 文件 进行 有 效 验证 ， 从 而 可 以 有 效 地 提高 系统 的 安全 性 
和 人 性 化 程度 ， 但 在 实际 应 用 中 ， 保 存 上 传 文件 时 需要 更 改 上 传 文件 名 或 对 上 传 文件 进行 加 密 操作 ， 这 时 
就 需要 使 用 适配器 类 的 addFilter0 方 法 为 上 传 增加 过 滤 规 则 。 在 具体 讲解 addFilter0 方 法 前 ， 首 先 需要 了 解 


常用 的 过 滤 规 则 有 哪些 ， 寺 


局 


其 中 表 25.3 所 示 为 Zend_File_Transfer Adapter_Http 适配器 类 常用 的 过 滤 规 则 。 
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表 25.3 Zend_File_Transfer_Adapter_Http 适配器 类 常用 的 过 滤 规 则 


说 
对 上 传 文件 进行 解密 过 滤 
对 上 传 文件 进行 加 密 过 滤 
将 上 传 的 文本 类 型 的 文件 中 的 英文 字符 转换 为 小 写 
将 上 传 的 文本 类 型 的 文件 中 的 英文 字符 转换 为 大 写 
对 上 传 文件 进行 重 命名 


使 用 addFilter0 方 法 为 上 传 文件 增加 验证 规则 一 般 需 要 为 该 方法 指定 3 个 参数 ， 第 1 个 参数 为 要 指定 的 
过 滤 规 则 名 称 ; 第 2 个 参数 为 数组 型 ， 用 于 为 该 过 滤 指 定 具 体 的 过 滤 规 则 ， 第 3 个 参数 用 来 指定 该 过 滤 规 
则 要 限定 的 文件 域名 称 。 

例如 ， 在 图 片上 传 时 使 用 Zend File_ Transfer Adapter Http 适配器 类 更 改 上 传 文件 名 为 ewName.gif 的 
实现 代码 如 下 : 


明 


Decrypt 
Encrypt 
LowerCase 


$adapter = new Zend_File_Transfer_Adapter_Http(); // 实 例 适配器 类 
$adapter->setDestination("./upfiles'); // 定 义 上 传 文件 保存 路 径 
$adapter->addFilter('Rename', array(target => 'newName.gif, 'overwrite' => true), "magename'); 。 // 改 文件 名 
$adapter->receive(); /执行 上 传 操作 


25.6 Zend Layout 网 站 布局 


通过 Zend_Layout 网 站 布局 ， 可 以 提高 网 站 页 面 的 重用 率 ， 便 于 网 站 内 容 的 更 新 和 维护 。 例 如 ， 在 一 个 
网 站 中 ， 每 个 页 面 中 都 可 能 包含 相同 的 头 、 尾 文件 ， 所 以 在 每 个 页 面 中 都 需要 编写 相同 的 头 、 尾 文件 。 如 
果 应 用 Zend_Layout 对 页 面 进行 布局 ， 将 页 面 的 头 、 尾 文件 定义 到 单独 的 模板 中 ， 这 样 在 创建 页 面 时 就 不 用 
再 编写 相同 的 头 、 尾 文件 代码 ， 只 要 调用 指定 的 头 、 尾 模板 文件 即 可 。 


25.6.1 Zend_Layout 概述 


在 Zend Framework 中 ， 使 用 Zend_Layout 组 件 实现 对 网 站 视图 层 页 面 的 布局 ， 从 而 可 以 有 效 提高 项 目 
的 可 维护 性 、 代 码 重 用 率 和 扩展 能 力 ， 使 用 Zend_Layout 对 网 站 布局 具有 以 下 优点 。 
建立 Zend Framework 的 MVC 框架 后 ， 可 以 自动 配置 Zend_Layout 的 解析 。 
能 够 更 好 地 实现 业务 逻辑 和 视图 层 的 分 离 。 
Zend Framework 中 允许 在 application.ini 文件 中 配置 布局 名 称 、 布 局 脚本 及 布局 脚本 路 径 。 
在 不 需 使 用 布局 的 动作 中 ， 可 以 通过 代码 取消 布局 。 
Zend Layonut 在 没有 建立 Zend Framework 的 MVC 框架 的 前 提 下 可 以 独立 使 用 。 


图 图 加 图 加 


25.6.2 Zend_Layout 使 用 方法 


以 往 在 使 用 PHP 开发 Web 应 用 时 ， 实 现 网 站 的 布局 是 使 用 文件 包含 函数 将 网 站 头 部 文件 、 尾 部 文件 及 
其 他 模块 包含 到 程序 中 。 而 使 用 Zend_Layout 实现 网 站 的 布局 , 只 需 简 单 的 配置 即 可 .下 面 以 Zend Framework 
的 MVC 框架 为 例 ， 讲 解 Zend_Layout 实现 站 点 布局 的 方法 和 步骤 。 
(1) 首先 需要 在 application.ini 文件 中 指定 布局 文件 的 保存 目录 和 布局 文件 名 。 其 代码 如 下 : 
resources.layout.layoutPath = APPLICATION_PATH "/layouts" 


@ 
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resources.layout.layout = "default" 

上 述 配 置 代码 指定 布局 文件 的 保存 目录 为 应 用 路 径 下 的 layouts 目录 ， 并 指定 该 目录 下 的 default.phtml 
文件 为 默认 布局 文件 。 这 里 需要 注意 ， 在 配置 网 站 布局 时 ， 可 以 不 指定 布局 文件 ， 但 需要 在 控制 器 的 动作 
方法 中 通过 如 下 代码 指定 。 

$this->_helper->layout->setLayout('default ); 

(2) 完成 application.ini 的 布局 配置 后 ， 就 可 以 建立 网 站 的 头 部 导航 〈headerphtml) 和 尾部 导航 
(footerphtml) 视图 文件 ， 为 了 能 够 通过 Zend View 对 象 的 render0 方 法 获取 到 这 两 个 文件 ， 需 要 将 
header.phtml 和 footer.phtml 文件 直接 保存 到 框架 的 scripts 目录 下 。 
(3) 完成 网 站 头 部 导航 和 尾部 导航 建立 后 ， 就 可 以 建立 布局 文件 defaultphtml， 根 据 步骤 (1) 中 的 配 
置 ，default.phtml 文件 被 保存 在 应 用 路 径 的 layouts 目录 下 。 其 关键 代码 如 下 : 
<html> 


<head> 
<!- 指 定 网 站 头 信息 -> 
</head> 
<body 
<!- 包 含 头 部 导航 一 > 
<?php echo $this->render('default_header.phtml)?> 
<!- 包 含 页 面 主体 部 分 --> 
<?php echo $this->layout()->content;?> 
< 包含 尾部 导航 --> 
<?php echo $this->render('default_footer. phtml)?> 
</body> 
</html> 
从 上 述 代 码 可 知 , 在 布局 文件 中 包含 网 站 的 头 部 文件 和 尾部 文件 是 通过 视图 对 象 的 render0 方 法 实现 的 ， 
包含 网 站 的 主体 内 容 是 通过 视图 对 象 的 layout0 方 法 返回 值 的 content 属性 实现 的 。 


25.6.3 ”Zend_Layout 应 用 实例 


为 了 能 够 更 加 清楚 地 掌握 Zend_Layout 组 件 实现 网 站 布局 的 配置 方法 ， 下 面 通过 具体 实例 进行 讲解 。 
例 25.6 使 用 Zend_Layout 实现 对 明日 IT 新 闻 网 的 页 面 布局 。( 实 例 位 置 : 光盘 \TMNInstance\2S\25.6) 
明日 IT 新 闻 网 是 一 个 专业 的 IT 新闻 类 网 站 , 在 网 站 的 首页 、 新闻 列表 页 和 新 闻 详 细 信 息 页 面 都 使 用 相 
同 的 网 站 头 部 和 尾部 导航 ， 为 了 提高 代码 的 重用 率 及 设计 统一 的 页 面 风格 ， 在 实施 网 站 布局 时 采用 
Zend_Layout 组 件 实现 。 
(1) 建立 Zend Framework 的 MVC 框架 结构 ， 具 体 创建 过 程 请 参见 本 章 第 25.2 节 的 内 容 ， 这 里 不 再 烤 述 。 
(2) 打开 application.ini 文件 ， 加 入 如 下 配置 代码 来 指定 布局 文件 的 保存 目录 。 
resources.layout.layoutPath = APPLICATION_PATH "layouts" 
(3) 建立 明日 IT 新 闻 网 的 首部 导航 视图 和 尾部 导航 视图 ， 其 效果 分 别 如 图 25.12 和 图 25.13 所 示 ， 并 
分 别 将 这 两 个 视图 文件 命名 为 headerphtml 和 footerphtml。 


算 起 李 鸭 喇 能 碟 的 到 


Ee 忆 W 中 


图 25.13 ”网 站 尾部 导航 
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(4) 在 网 站 布局 目录 下 建立 default.phtml 视图 文件 。 其 代码 如 下 : 
<IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/ 
xhtml1-transitional.dtd"> 
<html xmlns="http:/www.w3.org/1999/xhtml"> 
<head> 
<?php 
echo S$this->headMeta()->setHttpEquiv('content-type", text/html; charset=utf-8) 
->appendHttpEquiv('x-ua-compatible", "ie=7"); 
?> 
<?php echo $this->headLink()->setStylesheet($this->baseUrl('/css/style.css'))?> 
<style type="text/css"> 


.default-body {background:url(<?php echo Sthis->baseUrl(Vimg/bg.jpg')?>) 
repeat-x 0 0; background-color:#E7F6FB;} 

.default-top {width:980px; height:30px; clear:both;} 

.default-top .li1 {width:600px; height:30px; line-height:30px; text-align:left; 
color:#FFFFFF:; float:left;} 

.default-top .li2 {width:200px; height:30px; line-height:30px; text-align:right; 
color:#FFFFFF; float:right;} 

.default-main {width:980px; background-color:#FFFFFF; clear:both;} 

</style> 
</head> 


<body class="default-body"> 

<div class="default-top"> 

<ul> 
</ul> 

</div> 

<div class="default-main"> 
<?php echo $this->render('default_header.phtml')?> 
<?php echo $this->layout()->content;?> 
<?php echo $this->render('default_footer.phtml)?> 


</div> 
<div class="cell_h"></div> 
</body> 
</html> 
(5) 建立 IndexController 控制 器 ， 并 在 其 中 建立 indexAction 动作 。 其 代码 如 下 : 
<?php 


class IndexController extends Zend_Controller_Action{ 
public function indexAction(}{ 
Sthis->_helper->layout->setLayout('default’); /指定 布局 文件 为 default.phtml 


> 
} 
通过 上 面 步骤 即 可 实现 通过 Zend_Layout 组 件 对 网 站 进行 布局 ， 本 实例 的 运行 效果 如 图 25.14 所 示 。 


公 明 97 网 记名 相 全 光 避 能 A 入 纠 


图 25.14 明日 IT 新闻 网 的 网 站 布局 效果 
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25.7 Zend Paginator 分 页 


在 制作 网 站 的 数据 展示 模块 时 ， 如 果 数 据 量 较 大 ， 一 般 采用 分 页 显示 数据 的 方式 ，Zend_Framework 框 
架 提 供 的 Zend_Paginator 组 件 是 专门 用 于 数据 分 页 显示 的 ， 本 节 将 对 该 组 件 进行 介绍 。 


25.7.1 Zend_Paginator 简介 


Zend_Paginator 是 一 个 非常 优秀 的 数据 分 页 组 件 ， 该 组 件 采用 适配器 方式 ， 根 据 不 同 的 适配器 可 以 实现 
、Zend_Select 查询 结果 的 分 页 展示 。 该 组 件 具有 如 下 优点 。 

对 任意 数据 进行 分 页 ， 而 不 是 针对 关系 型 数据 库 。 

针对 需求 数据 进行 呈现 ， 而 不 是 全 部 数据 。 

不 强制 用 户 只 使 用 一 种 途径 呈现 数据 和 演 当 分 页 控件 。 

可 以 单独 使 用 (相对 Zend Framework 框架 本 身 ) 。 


对 数 


办 办 办 入 


25.7.2 ”Zend_Paginator 分 页 方法 


使 用 Zend_Paginator 组 件 实现 数据 的 分 页 显示 ,可 以 简化 操作 ,有 效 地 提高 程序 的 开发 效率 和 后 期 维护 
工作 。 下 面 讲解 使 用 Zend_Paginator 实现 数据 分 页 的 方法 。 
(1) 使 用 Zend_Paginator 组 件 实现 数据 分 页 ， 首 先 需要 确定 要 使 用 的 分 页 适配器 ， 常 用 的 分 页 适配器 
有 数组 适配器 和 Zend_Select 分 页 适配器 ， 其 中 数组 适配器 使 用 Zend_Paginator Adapter_ Array 类 指定 ， 
Zend_Select 查询 结果 的 分 页 显示 所 需 的 适配器 使 用 Zend_Paginator _ Adapter_ DbSelect 指定 。 例 如 ， 使 用 
Zend_Paginator_Adapter_Array 适配器 的 代码 如 下 : 
Sarray=array(); // 定 义 要 分 页 的 数组 
$adapter = new Zend_Paginator_Adapter_Array($array); /实例 化 适配器 
(2) 创建 完 分 页 适配器 后 ， 就 对 Zend_Paginator 分 页 类 进行 实例 化 并 指定 分 页 参数 。 该 过 程 的 代码 


如 下 : 
$paginator = new Zend_Paginator($adapter); /实例 化 Zend_Paginator 对 象 
$paginator->setPageRange(5) /指定 分 页 控制 器 面板 显示 的 页 码 个 数 
->setltemCountPerPage(20) /指定 每 页 显示 的 记录 数 
->setCurrentPageNumber(1); // 指 定 当前 显示 的 页 码 
Sthis->view->paginator = $paginator; // 将 分 页 类 赋值 给 视图 变量 


(3) 获得 Zend_Paginator 分 页 类 实例 化 对 象 后， 就 可 以 在 视图 层 通 过 传递 的 分 页 类 对 象 显示 分 页 数据 
和 分 页 控制 面板 ， 分 页 数据 通过 foreach 语句 循环 输出 。 例 如 ， 通 过 如 下 代码 显示 分 页 数据 中 ID 的 值 。 
foreach($this->paginator as $data) /遍历 分 页 类 对 象 


echo $date[id]; // 输 出 ID 的 值 
} 
(4) 通过 foreach 语句 循环 输出 指定 页 码 的 分 页 数据 后 ， 还 需要 制作 用 于 实现 分 页 控制 的 分 页 链接 面 
板 。 分 页 链接 控制 面板 文件 需要 放置 在 框架 的 scripts 目录 中 ， 该 文件 主要 用 于 生成 分 页 控制 面板 视图 ， 常 
用 的 页 面 属性 如 表 25.4 所 示 。 


@ 
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表 25.4 ”分 页 链接 控制 面板 文件 的 参数 说 明 


参数 说 有明 
first 首页 页 码 
firstltemNumber 当前 页 码 中 第 一 条 数据 在 完整 数据 集中 的 序号 (位 置 ) 
firstPageInRange 程序 运行 开始 时 实现 的 页 码 
current 当前 页 码 
currentItemCount 当前 页 数据 数 
itmCountPerPage 本 页 最 多 数据 数量 
last 最 后 页 码 
lastItemNumber 当前 页 码 中 最 后 一 条 数据 在 完整 数据 集中 序号 (位置 ) 
next 下 一 页 页 码 
pageCount 总 页 数 
pageInRange 现实 页 码 数 组 
previous 上 一 页 页 码 
totalItemCount 总 数据 数量 
根据 表 25.4 可 知 ， 在 分 页 链接 控制 面板 文件 中 可 以 通过 如 下 代码 显示 “最 后 一 页 ”导航 链接 。 


<a href="<?php echo $this->baseUrl(/newsjlist?page='.$this->last)?>"> 最 后 一 页 </a> 
(5) 建立 完 分 页 链接 控制 面板 页 后 ， 还 需要 在 分 页 视图 中 使 用 视图 对 象 的 paginationControl0 方 法 导入 

分 页 面板 控制 页 面 。 实 现 导入 分 页 面板 页 面 的 代码 如 下 : 

<?php echo S$this->paginationControl($this->paginator, 'Sliding'", '\ist_news_pagination_control.phtml'， 

array(flag'=>$this->flag, 'order=>$this->order))?> 

paginationControl0 方 法 一 般 有 4 个 参数 ,其 中 第 1 个 参数 用 于 指定 分 页 控制 面板 关联 的 Zend_Pageinator 
对 象 ， 第 2 个 参数 用 于 指定 控制 面板 样式 ， 其 中 常用 的 样式 及 说 明 如 表 25.5 所 示 。 第 3 个 参数 用 来 指定 分 
页 控制 面板 的 页 面 文件 名 称 ， 最 后 一 个 参数 为 数组 型 ， 用 于 向 分 页 面板 文件 传递 视图 变量 。 


表 25.5 分 页 样式 详细 说 明 


返回 每 个 页 码 ， 这 是 下 拉 菜 单 跳 转 中 的 分 页 
Elastic 类 似 于 Google 分 页 样式 
Jumping 跳 转 到 最 后 一 页 后 ， 返 回 到 首页 页 码 


Sliding 类 似 于 Yahoo 分 页 样式 ， 以 当前 页 为 中 心 《推荐 ) 
25.7.3 Zend_Paginator 分 页 应 用 


通过 上 节 的 学 习 可 以 掌握 使 用 Zend_Paginator 实现 数据 分 页 的 基本 思路 和 方法 , 本 节 将 通过 具体 实例 讲 
解 如 何 使 用 Zend_Paginator 实现 分 页 。 

例 25.7 使 用 Zend Paginator 实现 对 明日 IT 新 闻 网 的 新 闻 主题 的 分 页 显示 。【〔 实 例 位 置 ， 光盘 \ 
TMWY2525.7) 

在 制作 明日 IT 新 闻 网 的 新 闻 主 题 展 示 页 面 时 ， 使 用 Zend_Paginator 组 件 实现 主题 的 分 页 显示 ， 下 面具 
体 讲 解 该 模块 的 制作 方法 。 

(1) 建立 Zend Framework 的 MVC 框架 结构 ， 具 体 创建 过 程 请 参见 本 章 第 25.2 节 的 内 容 ， 这 里 不 再 獒 述 。 

(2) 建立 新 闻 表 模型 Model DbTable News 类 ， 并 在 该 类 中 建立 用 于 实现 新 闻 分 页 的 findByPage0 方 
法 。applicationmodels\DbTable News.php 文件 的 代码 如 下 : 

a 
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public function findByPage ($flag, $orderlndex, $page = 1, $itemCountPerPage = 10, $pageRange = 5) 
. 
$select = $this->getAdapter()->select(); /生成 select 对 象 
S$select->from(S$this->_name); // 指 定 要 查询 的 表 名 
if ($flag !== 'al){ /指定 查询 条 件 
if ($flag === 区 ){ 
S$where = 'istodaycommend = true'; 
}else{ 
S$where = $this->getAdapter()->quotelnto(flag = ?', $flag); 


下 
S$select->where($where); 


} 
if ($orderindex == 0) { /排序 方式 
$order = 'addtime desc'; 
} elseif ($orderlIndex == 1){ 
S$order = 'addtime'; 
} elseif ($orderlndex == 2){ 
S$order = 'browse desc' 
} elseif ($orderlndex == 3) { 
S$order = 'browse'; 


》 
$select->order($order); // 排 列 顺序 
$paginatorAdapter = new Zend_Paginator_Adapter_DbSelect($select); /实例 并 构建 Zend_Paginator 对 象 
$paginator = new Zend_Paginator($paginatorAdapter); 
S$paginator->setPageRange($pageRange)->setltemCountPerPage($itemCountPerPage)->setCurrentPageNum 
ber($page); // 指 定 分 页 参数 
return $paginator; /| 返回 Zend_Paginator 对 象 
3 
上 述 代 码 首 先 构 建 Zend_Select 查询 对 象 ， 然 后 使 用 Zend_Paginator Adapter_DbSelect 分 页 适配器 生成 
用 于 对 Zend_Select 查询 结果 进行 分 页 的 适配器 对 象 ， 同 时 实例 化 Zend_Paginator 分 页 类 对 象 并 指定 分 页 参 
数 ， 最 后 返回 分 页 类 对 象 。 
(3) 在 IndexController 控制 器 的 indexAction 动作 中 ， 调 用 新 闻 表 模型 的 findByPage0 方 法 ， 并 将 返回 
结果 赋值 给 视图 变量 ， 以 便 在 视图 层 应 用 。application\controllers\IndexController.php 文件 的 代码 如 下 : 
public function indexAction () 


// 接 收 参数 
if($this->_request->getParam(’flag')=="" || S$this->_request->getParam('order'’)=="" ||$this->_request-> 
getParam('page')=="" 

S$flag ="al 
S$order = 2; 
S$page = 1; 

jelsef 
$flag = Sthis->_request->getParam(flag '); 
$order = $this->_request->getParam(order); 
$page = $this->_request->getParam('page'); 


} 

// 分 页 参数 

S$itemCountPerPage = 10; 

S$pageRange = 5; 

// 视 图 变量 赋值 

Sthis->view->flag = $flag; 

S$this->view->order = $order; 

// 新 闻 分 页 

S$paginator = $this->_news->findByPage($flag, $order, $page, $itemCountPerPage, $pageRange); 
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Sthis->view->paginator = $paginator; 
Sthis->_helper->layout->setLayout("default"); 
} 
(4) 在 视图 层 ， 使 用 foreach 循环 语句 输出 新 闻 主题 信息 。application\views\scripts\index\index.phtml 
文件 的 代码 如 下 : 
<?php foreach ($this->paginator as $news):?> 
<div class="content"> 
<div class="title"> 
-<a href="#" class="a25"><?php echo S$this->escape($news['title])?></a> 
</div> 
<div class="description"> 
<?php echo trim($this->escape($this->substr($news['uncontent], 220)))?> <a href="#" class="a3"> 
<nobr>&lt; 详 细 &gt;</nobr></a> 
</div> 
<div class="msg"> 
来 源 : <font color="#888888"><?php echo $this->escape($news[source])?></font> | 作者 : <font 
color="#888888"><?php echo S$this->escape($news['author])?></font> | 时 间 : <font color="#888888"><?php 
echo substr($news['addtime], 0, 16)?></font> | 浏览 : <font color="#888888"><?php echo $news[browse']?> 
</font>&nbsp; 次 
</div> 
</div> 
<?php endforeach;?> 
(5) 在 index.phtml 页 面 中 使 用 paginatorControl0 方 法 调用 分 页 控制 面板 。 其 代码 如 下 : 
<?php echo S$this->paginationControl($this->paginator, 'Sliding', ‘list_news_pagination_control.phtml', 
array('flag'=>$this->flag, 'order=>$this->order))?> 
(6) 建立 分 页 控制 面板 。application\views\scripts\list_news_pagination_control.phtml 文件 的 代码 如 下 : 
<div style="width:280px; height:20px; line-height:20px; color:#0257AE; font-weight:bold; text-align:center; 
border:1px solid #92C7ED; background-color:#FAFEFF; margin-right:8px; float:left,"> 
每 页 &nbsp;<font style="color:#999999; font-family:Arial; font-size:12px;"><?php echo S$this->itemCountPerPage?> 
</font>&nbsp; 条 / 共 &nbsp;<font style="color:#999999; font-family:Arial; font-size:12px;"><?php echo 
S$this->totalltemCount?></font>&nbsp; 条 第 &nbsp;<font style="color:#999999; font-family:Arial; font-size:12px;"> 
<?php echo S$this->current?></font>&nbsp; 页 / 共 &nbsp;<font style="color:#999999; font-family:Arial; 
font-size:12px;"><?php echo S$this->pageCount?></font>&nbsp; 页 
</div> 
<?php if($this->totalltemCount > 0):?> 
<?php if($this->current > 1):?> 
<div style="width:40px; height:20px; line-height:20px; border:1px solid #92C7ED; background-color:<?php 
if($this->current==1):?>#1A75D2;<?php else:?>#FAFEFF;<?php endif;?> font-weight:bold; float'left margin- 
right:8px;"> 
<a href="<?php echo S$this->baseUrl('/list-".$this->flag.'-".$this->order.'-1.html")?>" class="<?php 
if($this->current==1):?>a4<?php else:?>a3<?php endif?>"> 首 页 </a> 
</div> 
<?php endif ?> 
<?php foreach (S$this->pagesInRange as $page):?> 
<div style="width:20px; height:20px; line-height:20px; border:1px solid #92C7ED; background-color:<?php 
if($this->current==$page):?>#1A75D2;<?php else:?>#FAFEFF;<?php endif;?> font-family:Arial; font-style:italic; 
font-weight:bold; float:left; margin-right:8px;"> 
<a href="<?php echo S$this->baseUrl('/list-". $this->flag.'-".$this->order.'-'.$page.".html'")?>" class="<?php 
if($this->current==$page):?>a4<?php else:?>a3<?php endif?>"><?php echo $page?></a> 
</div> 
<?php endforeach;?> 
<?php if($this->pageCount > $this->pageRange && $this->pageCount!=$this->current):?> 
<div style="width:40px; height:20px; line-height:20px; border:1px solid #92C7ED; background-color:<?php 
if($this->current==$this->last):?>#1A75D2;<?php else:?>#FAFEFF;<?php endif?> font-weight:bold; float:left; 
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margin-right:8px;"> 
<a href="<?php echo Sthis->baseUrl(/list-. $this->flag.-'. $this->order.-'. $this->last.'.html')?>" class="<?php if 
($this->current==$this->last):?>a4<?php else:?>a3<?php endif;?>"> 尾 页 </a> 
</div> 
<?php endif?> 
<?php endif;?> 


以 上 为 明日 开 新 闻 网 新 闻 主题 循环 输出 的 实现 过 程 ， 运 行 本 实例 ， 效 果 如 图 25.15 所 示 。 


从 明日 37 新 闻 列 从 起 要 的 司 能 找 的 到 


| 


图 25.15 明日 IT 新 闻 网 新 闻 主 题 分 页 展示 


25.8 实 战 
句 1 视频 讲解 :光盘 \TM\Video\ 第 25 章 \ 实 战 .exe 
25.8.1 Zend_Paginator 实现 数据 分 页 显示 


例 25.8 在 本 实例 中 , 应 用 Zend_Paginator 对 数据 库 中 数据 进行 分 页 显示 , 其 运行 结果 如 图 25.16 所 示 。 
(实例 位 置 ， 光 盘 \TM\Instance\25\25.8) 


捉 # 春 天 股 关 @ 心 二 


25.16 Zend_Paginaator 分 页 显示 


有 具体 步骤 如 下 。 
(1) 搭建 Zend Framework MVC 环境 。 
(2) Zend_Config 配置 站 点 初始 参数 。 
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(3) 在 application\controllers\IndexController.php 中 加 入 如 下 代码 : 
S$pageNumber = 5; 

@$paginator = Zend_Paginator::factory($this->view->studentsModel); 
$paginator->setltemCountPerPage($pageNumber); 
$paginator->setCurrentPageNumber($this->_getParam(page')); 
$paginator->setPageRange(5); 

四 Zend_Paginator:setDefaultScrollingStyle('Sliding'); 
$paginator->setView($this->view); 

S$this->view->studentsModel = $paginator; 
©sthis->view->paginator = $paginator; 
Sthis->render(); 


CN 
@ Zend Paginator::factory0 为 分 页 器 的 工厂 方法 ， 参 数 为 数组 中 读 取 出 的 数据 (数组 )。 
@ Zend Paginator::setDefaultScrollingStyle('Sliding) 调 用 页 码 样 式 。 
@ $this->view->paginator 将 分 页 传 给 V 层 。 


(4) 在 application\views\scripts\index 文件 下 创建 pagelistphtml 文件 ， 建 立 所 需要 的 分 页 页 码 。 其 代码 
如 下 : 
<?php if ($this->pageCount): ?> 
<div class="paginationControl"> 
<!-- Previous page link —> 
<?php if (isset($this->previous)): ?> 
<a href="<?php echo S$this->url(array('controller'=>'index', 'action'=>'index", ‘page'=>$this->previous)); ?>">< 
上 一 页 </a> | 
<?php else: ?> 
<span class="disabled">< 上 一 页 </span> | 
<?php endif; ?> 
<!-- Numbered page links --> 
<?php foreach ($this->pagesinRange as $page): ?> 
<?php if ($page != $this->current): ?> 
<a href="<?php echo S$this->url(array('controller'=>'index', 'action'=>'index', 'page'=>$page)); ?>"><?= 
S$page; ?></a> | 
<?php else: ?> 
<?= $page; ?> | 
<?php endif; ?> 
<?php endforeach; ?> 
<!-- Next page link --> 
<?php if (isset($this->next)) ?> 
<a href="<?php echo $this->url(array('controller'=>'index', 'action'=>'index', ‘page'=>$this->next)); ?>"> 下 一 
页 ></a> 
<?php else: ?> 
<span class="disabled"> 下 一 页 ></span> 
<?php endif; ?> 
</div> 
<?php endif; ?> 


25.8.2 ”Zend Framework 用 户 身 份 验证 


例 25.9 使 用 Zend Framework 框架 实现 用 户 身份 的 验证 主要 应 用 Zend_Auth 组 件 。 明 日 IT 新 闻 网 的 
用 户 登 录 模 块 就 是 应 用 Zend_Auth 组 件 对 用 户 身份 进行 验证 。〔 实 例 位 置 光盘 \TMNInstance\2S\2S.9) 


其 具体 步骤 如 下 。 
@ 


(1) 建立 Zend Framework 的 MVC 框架 结构 。 
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(2) 建立 用 户 登 录 表 单 。 该 表单 中 元 素 包括 用 户 昵称 文本 框 、 用 户 密码 文本 框 和 验证 码 文本 框 ， 实 现 
该 过 程 的 代码 主要 由 HTML 语句 的 表单 相关 标签 组 成 ， 具 体 实 现代 码 请 详 见 本 书 附 带 光 盘 ， 其 运行 效果 如 
图 25.17 所 示 。 


公 7 


Pe 


中 1999-2010 归 日 TW 届 册 


图 25.17 明日 IT 新 闻 网 的 登录 页 面 


(3) 在 用 户 表 模型 中 建立 isValid0 方 法 , 用 于 验证 登录 信息 是 否 正 确 , 正确 则 返回 tue, 反之 返回 false。 
applicationmodels\DbTable\News.php 文件 的 关键 代码 如 下 : 
public function isValid ($netname, $password) 


// 构 建 验证 适配器 

$authAdapter = new Zend_Auth_Adapter_DbTable($this->getAdapter()); 
$authAdapter->setTableName($this->_name)->setldentityColumn('netname')->setCredentialColumn('password' 
)->setldentity($netname)->setCredential($password); 

1/ 获取 Zend_Auth 对 象 并 验证 

$auth = Zend_Registry::get('auth'); 

return $auth->authenticate($authAdapter)->isValid(); 


(4) 当 用 户 在 用 户 登录 表单 中 输入 用 户 信息 并 单 击 “ 登 录 ”按钮 后 , 登录 信息 将 被 提交 到 UserController 
控制 器 的 loginAction 动作 中 验证 用 户 的 登录 信息 是 否 正确 applicationmodulesvdefaultvcontrollers\UserController 
.php 文件 的 代码 如 下 : 

if ($this->_request->isPost()) { 
// 接 收 用 户 名 和 密码 
S$netname = trim($this->_request->getParam('netname')); 
S$password = trim($this->_request->getParam('password')); 
// 对 用 户 身份 进行 验证 
if ($this->_user->isValid($netname, md5($password))) { 
Surl = $this->_request->getParam('url"); 
if ($url == null) { 
// 定 位 到 登录 成 功 页 面 
Sthis->_redirect('/user/success/flag/login'); 
}else{ 
Surl = str_replace('@", '?', str_replace('$", '&', $url)); 
Sthis->_redirect($url); 


} 
exit(); 
}else{ 


$errMsg = ' 登 录 昵 称 或 密码 输入 有 误 ， 请 重新 登录 !"; 
Sthis->view->errMsg = $errMsg; 


全 
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S$this->view->netname = $netname; 


} 
25.8.3 Zend_Mail 发 送 邮 件 


例 25.10 Zend Framework 通过 Zend Mail 组 件 发 送 邮件 ,邮件 在 Zend_Mail 里 通过 默认 的 Zend_Mail_ 
Transport_Sendmail 和 Zend_Mail _ Transport _ Smtp 来 发 送 。 在 本 实例 中 通过 Zend_Mail Transport_Smtp 来 发 
送 邮 件 。〔 实 例 位 置 ， 光盘 \TM\Instance25W25.10) 

发 送 邮件 应 用 Zend_Mail Transport_Smtp(S$ip)，$ip 地 址 为 服务 器 IP 地 址 。Zend_Mail 发 送 邮件 有 4 个 
基本 属性 ， 如 表 25.6 所 示 。 


表 25.6 ”Zend_Mail 发 送 邮 件 的 4 个 基本 属性 


addTo($option.$to) 收 件 人 地 址 ，$option 为 发 件 人 地 址 ，$to 为 邮件 中 发 件 人 地 址 


serFrom($option.$from) 发 件 人 地 址 ，$option 为 收 件 人 地 址 ，$from 为 邮件 中 收 件 人 地 址 
setSubject($subject) 
setbodyText($body) 


其 操作 步骤 如 下 。 
(1) 搭建 Zend Framework MVC 环境 。 
(2) Zend_Config 配置 站 点 初始 参数 。 
(3) 在 applicationmodels\DbTable 文件 下 创建 MailForm.php 文件 ， 创 建 4 个 表单 : 收 件 人 、 发 件 人 、 


标题 和 内 容 。 其 代码 如 下 : 
class Model_DbTable_MailForm extends Zend_Form 


public function __ construct($option = null) 


parent::_construct($option); 
S$this->setName('form1"); 
//form 控制 器 需要 知道 有 哪些 内 容 
$to = new Zend_Form_Element_Text('to'); 
$to->setLabel(' 收 件 人 ') 
->setRequired(true) 
->addFilter('StripTags') 
->addFilter('StringTrim') 
->addValidator(NotEmpty ); 
S$from = new Zend_Form_Element_Text(from ); 
S$from->setLabel(' 发 件 人 ') 
->setRequired(true) 
->addFilter('StripTags') 
->addFilter('StringTrim') 
->addValidator('NotEmpty'); 
$subject = new Zend_Form_Element_Text('subject'); 
$subject->setLabel(' 标 题 ') 
->setRequired(true) 
->addFilter('StripTags') 
->addFilter('StringTrim') 
->addValidator(NotEmpty ); 
$content = new Zend_Form_Element_Textarea('content'); 
$content->setLabel( 内容 ') 
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->setRequired(true) 
->addFilter('StripTags') 
->addFilter('StringTrim’) 
->addValidator('NotEmpty’); 
$submit = new Zend_Form_Element_Submit('submit’); 
$submit->setAttrib("id','submitbutton"); 
Sthis->addElements(array($to, $from, $subject, $content, $submit)); 
b 
(4) 在 application\controllers 文件 下 创建 mdexControllerphp， 确 定 使 用 类 及 服务 器 IP 地 址 ， 实 例 化 
Zend_Mail， 填 入 表单 元 素 ， 发 送 邮件 。 其 代码 如 下 : 
public function indexAction() 
{ 


Sform = new Model_DbTable_MailForm(); 
$form->submit->setLabel(' 发 送 "); 
Sthis->view->form = $form; 

if ($this->_request->isPost()) 

{ 


SformData = $this->_request->getPost(); 

if ($form->isValid($formData)) 

{ 
Str = new Zend_Mail_Transport_Smtp("192.168.1.247"); 
Smail = new Zend_Mail(); 


S$mail->addTo('‘cym3100@163.com',$form->getValue('to"); 
Smail->setFrom('‘cym3100@163.com',$form->getValue('from’)); 
Smail->setSubject($form->getValue('subject')); 
Smail->setBodyText($form->getValue('content')); 


Smail->send($tr); 
Sthis->_redirect('/"); 

jelse 

{ 
Sform->populate($formData); 


} 
有 
(5) 在 application\views\scripts\index 文件 下 创建 index.phtml， 在 该 页 面 中 调用 Form 表单 。 其 代码 如 下 : 
<?php echo S$this->form; ?> 
运行 结果 如 图 25.18 所 示 。 
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图 25.18 Zend_Mail 发 送 邮 件 
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oh, 
SO 说明 Zend_Mail 发 送 邮 件 需要 实例 化 Zend Mail Transport Smtp($ip), 这 个 他 地 址 必须 为 服务 器 IP。 
AddTo0 和 AddFrom0 中 的 第 一 个 参数 必须 是 服务 器 中 的 一 个 邮箱 名 称 。 


25.8.4 Zend_Form 制作 用 户 注册 表单 


例 25.11 Zend Form 是 Zend Framework 自 带 的 组 件 ， 为 Web 程序 中 简化 表单 创建 和 处 理 ， 它 可 以 将 
输入 的 元 素 过 滤 和 校 验 ， 还 可 以 对 表单 元 素 进 行 排序 、 解 析 。 运 行 本 实例 将 看 到 完整 的 Zend Form 用 户 注 
册 表 单 。 (实例 位 置 : 光盘 \TMNInstance\2S\25.11) 

具体 实现 步骤 如 下 。 

(1) 搭建 Zend Framework MVC 环境 。 
(2) Zend_Config 配置 站 点 初始 参数 。 
(3) 在 application\models\DbTable 文件 下 创建 StudentForm.php 文件 ， 建 立 Zend_Form, 确定 数据 库 表 
名 ， 设 置 表单 元 素 。 其 代码 如 下 : 
class Model_DbTable_StudentForm extends Zend_Form 
下 


publicfunction __ construct($option = null) 
{ 


parent::_construct($option); 
Sthis->setName('tb_students’); 
Sid = new Zend_Form_Element_Hidden('id"); 
Snum = new Zend_Form_Element_Text('num'); 
Snum->setLabel(' 学 号 ') 
->setRequired(true) 
->addFilter('StripTags') 
->addFilter('StringTrim') 
->addValidator('NotEmpty’); 
$name = new Zend_Form_Element_Text(name'); 
$name->setLabel(' 姓 名 ') 
->setRequired(true) 
->addFilter('StripTags') 
->addFilter('StringTrim') 
->addValidator('NotEmpty’); 
$age = new Zend_Form_Element_Text('age'); 
$age->setLabel(' 年 龄 ') 
->setRequired(true) 
->addFilter('StripTags') 
->addFilter('StringTrim') 
->addValidator('NotEmpty’); 
$class = new Zend_Form_Element_Text('class'); 
$class->setLabel(' 班 级 ) 
->setRequired(true) 
->addFilter('StripTags') 
->addFilter('StringTrim') 
->addValidator(NotEmpty ); 
$submit = new Zend_Form_Element_Submit(submit); 
$submit->setAttrib("id','submitbutton"); 
S$this->addElements(array($id,$num,$name,$age, $class,$submit)); 
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(4) 在 application\controllers 文件 下 创建 IndexController.php， 建 立 Form， 并 以 POST 方式 得 到 表单 中 
元 素 ， 存 入 数据 库 中 ， 返 回 本 页 面 。 其 代码 如 下 : 
public function indexAction() 
{ 
$form = new Model_DbTable_StudentForm(); 
$form->submit->setLabel(' 注 册 '); 
Sthis->view->form = $form; 
if ($this->_request->isPost()) 


| 
SformData = $this->_request->getPost(); 
if ($form->isValid($formData)) 
{ 
$students = new Model_DbTable_Students(); 
Srow = $students->createRow(); 
S$row->num = $form->getValue('num'); 
Srow->name = $form->getValue('name'); 
Srow->age = $form->getValue('age’); 
Srow->class = $form->getValue('class'); 
Srow->save(); 
Sthis->_redirect('/"); 
jelse 
$form->populate($formData); 
用 
} 


(5) 在 application\views\scripts\index 下 建立 index.phtml 文件 , 文件 中 只 需要 调用 控制 层 中 的 显示 form 
即 可 实现 。 代 码 如 下 : 
<?php echo $this->form; ?> 


多 站 对 于 Model 层 中 的 文件 ,如 果 类 的 名 称 前 加 入 DbTable, 则 相应 地 在 models 文 件 下 建立 DbTable 
文件 ， 将 类 文件 建 在 下 面 ; 反之 则 不 必 建 立 。 


运行 结果 如 图 25.19 所 示 。 
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图 25.19 Zend Form 用 户 注册 表单 
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25.9 小 结 


本 章 主要 介绍 Zend Framework 框架 的 MVC 环境 搭建 、 配 置 、 框 架 结构 、 分 页 ， 并 通过 实例 ， 对 Zend 
Framework 的 各 种 应 用 进行 了 讲解 ， 以 此 来 增加 读者 对 Zend Framework 的 理解 。 希 望 通过 本 章 的 学 习 ， 读 
者 能 够 掌握 Zend Framework 技术 ， 并 将 其 灵活 地 运用 到 实际 的 网 站 开发 中 。 


25.10 ”学 习 成 果 检验 


1. Zend_Db_Table 如 何 执 行 添加 、 删 除 和 修改 操作 ?实例 位 置 ， 光盘 \TM\Instance\25\25.12) 
2. 如 何 比较 缓存 日 期 与 本 地 日 期 ? (实例 位 置 ， 光盘 \TMNInstance\2S\25.13) 


#2 Os 


综合 实例 (五 ) 一 一 电子 商务 网 站 


( 铝 ! 视频 讲解 : S3 分 钟 ) 


随 着 PC 机 (个 人 计算 机 ) 的 发 展 和 互联 网 的 普及 ， 电 子 商务 从 
报 文 时 代 进 入 到 了 Internet 时 代 ， 并 逐渐 被 大 众 所 了 解 和 接受 。 电 子 
商务 (Electronic Commerce， 简 称 EC)， 是 目前 发 展 较 快 的 一 种 商务 
模式 。 迄 今 为 止 ， 不同 领域 的 人 对 EC 的 理解 各 有 不 同 。 简 单 地 说 ， 
EC 是 一 种 基于 Internet， 利 用 计算 机 硬件 、 软 件 等 现 有 设备 和 协议 进 
行 各 种 商务 活动 的 方式 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

WI 了 解 如 何 分 析 并 设计 数据 库 

WH 了 解 Ajax 无 刷新 验证 技术 

mm 了 解 使 用 JS 脚本 获取 、 输 出 标签 内 容 

mH 熟悉 在 新 窗口 使 用 session 

让 ”熟悉 smarty 模板 配置 类 文件 

WI 熟悉 分 页 技术 

Mm 熟悉 购物 车 模块 概述 


第 26 章 ”综合 实例 (五 ) 一 一 电子 商务 网 站 


26.1 电子 商务 网 站 概述 


20 世纪 90 年 代 , 互联 网 的 蓬勃 发 展 为 企业 提供 了 一 个 全 新 的 机 遇 。 企业 网 站 、 电子 商务 成 为 热门 话题 。 
其 中 ， 电 子 商务 更 是 关系 到 经 济 结构 、 产 业 升 级 和 国家 整体 经 济 竞争 力 。 为 此 ， 我 国 已 经 将 发 展 电子 商务 
列 为 信息 化 建设 的 重要 内 容 并 努力 创造 条 件 ， 积 极地 推进 电子 商务 的 发 展 。 

据 美国 在 线 (AOL) 和 Henley Centre 联合 进行 的 一 项 调查 显示 : 国外 有 80% 的 受 调查 者 会 选择 网 上 购 
物 或 寻求 帮助 ，10% 的 受 调查 者 会 选择 熟悉 的 品牌 或 厂商 来 购买 。 而 在 国内 ， 自 1997 年 拉 开 了 电子 商务 的 
序幕 ， 短 短 的 10 年 里 ， 全 国 已 有 4 万 家 商业 网 站 ， 几 乎 每 天 都 有 新 的 网 站 诞生 ， 厂 商 所 在 地 也 从 上 海 、 广 
州 、 深 圳 等 沿海 发 达 地 区 扩展 到 全 国 各 地 区 。 


26.2 系统 分 析 
区 Nd 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 系 统 分 析 .exe 


26.2.1 系统 目标 


根据 客户 提供 的 需求 和 对 实际 情况 的 考察 与 分 析 ， 该 电子 商务 应 该 具备 如 下 特点 。 
首页 设计 要 能 够 吸引 用 户 的 目光 ， 整 个 页 面 要 以 简洁 为 主 ， 突 出 重点 。 

可 操作 性 强 ， 避 免 复杂 的 、 有 异议 的 链接 。 

浏览 速度 快 ， 尽 量 避 免 长 时 间 打 不 开 页 面 的 情况 发 生 。 

商品 信息 部 分 有 实物 图 例 ， 图 像 清楚 、 文 字 醒 目 。 

详细 的 商品 查询 功能 ， 可 以 通过 商品 的 各 个 属性 来 搜索 。 

详细 的 流程 介绍 ， 从 浏览 商品 到 购买 结账 ， 各 个 步骤 之 间 的 联系 最 好 能 以 图 例 来 说 明 。 
提供 在 线 咨询 。 

后 台 可 以 对 用 户 信息 和 商品 信息 进行 详尽 的 查看 和 管理 。 

订单 管理 。 

易 维 护 ， 并 提供 二 次 开发 支持 。 


26.2.2 ”功能 流程 结构 


因 因 办 因 办 国共 办 办 罗 


电子 商务 平台 分 前 台 系 统 和 后 台 系统 。 下 面 分 别 给 出 前 、 后 台 的 系统 功能 结构 图 。 电 子 商务 前 台 系 统 
功能 结构 如 图 26.1 所 示 。 

电子 商务 后 台 系统 功能 结构 图 如 图 26.2 所 示 。 
26.2.3 程序 预览 


电子 商务 网 站 由 多 个 功能 模块 组 成 ， 为 了 让 读者 对 本 系统 有 个 初步 的 了 解 和 认识 ， 下 面 列 出 几 个 典型 
功能 的 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 
@ 
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电子 商务 前 台 系统 功能 结构 图 
最 | [ 瞪 | 皂 | 击 | 诬 | 国 | 朵 | [网 | fy| 版 | 庙 | 撕 
新 | | 着 | | 站 | | 品 | | 找 | | 户 | | 户 | | 物 | | 单 | | 情 | | 城 | | 回 
商 | | 商 | | 商 | | 分 | | 雷 | | 注 | | 登 | | 流 | | 查 | | 链 | | 公 | | 密 
品 | | 品 | | 品 | | 类 | | 品 | | 开 | | 录 | | 程 | | 询 | | 接 | | 告 | | 码 
查 购 
看 | | 入 | | 买 查 区 着 | | 注 
购 购 | | 
商 商 信 | | 信 销 
品 | | 物 | | 品 物 
J 的 轿 回 中 | | 的 
图 26.1 电子 商务 前 台 系统 功能 结构 图 
电子 商务 后 台 系 统 功能 结构 图 


类 商 用 公 链 
型 昌 户 省 接 
和 管 管 管 管 
理 理 理 理 理 

添 | 读 查 | | 涛 | | 埋 上 会 | | 添 | | 刁 | 计 

加 | | 和 看 | | 如 | | 知 和 员 | | 如 天 各 | 区 

类 | | 类 商 | | 商 | 和 外 管 | | 会 | | 公 | 链 | | 链 

型 | | 刚 品 | | 品 | | 总 - 理 | | 告 | | 告 | 接 | | 

fn 人 
修 | | 出 | 修 | | 到 | | 处 | | 开 | | 党 | 各 | 本 | | 查 | | 四 | 陈 | | 修 | | 出 
改 | | 除 || 改 | | 除 | 恒 加 | | 党 | 隆 | | 看 || 除 | 结 | 和 除 改 | | 除 
类 | | 类 || 疝 | | 高 | 和 | | 订 | | 符 | | 答 | | 管 | | 会 || 会 | | 会 链 | | 链 
别 | 别 吕 | 品 | | 吕 | | 症 | | 融 | 到 | | 融 | | 员 || 员 [| 员 | 。 购 接 | | 接 


26.2 ”电子 商务 后 台 系统 功能 结构 图 


明日 购物 商城 系统 主页 如 图 26.3 所 示 ， 展 示 网 站 的 部 分 最 新 商品 、 热 门 商品 、 推 荐 商品 以 及 网 站 的 最 
新 公告 和 会 员 登 录 窗 口 。 


26.3 ”明日 购物 商城 首页 
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推荐 商品 展示 页 面 如 图 26.4 所 示 ， 分 页 展示 网 站 的 所 有 推荐 商品 。 
购物 车 页 面 如 图 26.5 所 示 ， 展 示 会 员 在 本 站 购买 的 商品 。 


壬 的 购物 千 
两 品名 称 购买 的 各 市 场 价格 会员 从 阁 折扣 这 合计 
避 [3 下 ] 3 dg.2 日 1047.0 
三 雪 罗 相机 [mw | 1886 1699.2 9 B496 
万。 法术 机 [| 3866 2393.4 日 388.4 
厂 了 六 搞 El] 8 4399.2 9 4399,2 
产品 5 个 基 II 十 苦 1 贾 上 23 页 起 页 上 -页 下 -- 素 民 T 主 这 5 和 “3 共计 ，163qz,2 元 
图 26.4 推荐 商品 展示 图 26.5 购物 车 


用 户 注册 页 面 如 图 26.6 所 示 ， 完 成 用 户 注册 信息 的 填写 和 验证 操作 。 


El 
: ED 二 国 swzn 
加 


图 26.6 用 户 注册 页 面 


26.3 数据库 设计 


无 论 是 什么 系统 软件 ， 其 最 根本 的 功能 就 是 对 数据 的 操作 与 使 用 。 所 以 ， 一 定 要 先 做 好 数据 的 分 析 、 
设计 与 实现 ， 然 后 才 实现 对 应 的 功能 模块 。 


26.3.1 数据 库 分 析 


根据 需求 分 析 和 系统 的 功能 流程 图 ， 找 出 需要 保存 的 信息 数据 〈 也 可 以 理解 为 现实 世界 中 的 实体 ) ， 


_- 国 
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并 将 其 转化 为 原始 数据 〈 属 性 类 型 ) 形式 。 这 种 描述 现实 世界 的 概念 模型 ， 可 以 使 用 E-R 图 来 表示 ， 也 就 
是 实体 -联系 图 。 最 后 将 E-R 图 转换 为 关系 数据 库 。 这 里 重点 介绍 几 个 E-R 图 。 

1. 会 员 信息 实体 

会 员 信息 实体 包括 编号 、 用 户 名 、 密 码 、E-mail、 身 份 证 号 、 联 系 电话 、QQ 号 、 密 码 保护 、 密 码 答案 、 
邮编 、 注 册 时 间 、 真 实 姓名 等 属性 。 会 员 信息 实体 E-R 图 如 图 26.7 所 示 。 

2. 商品 信息 实体 

商品 信息 实体 包括 编号 、 名 称 、 上 市 时 间 、 添 加 日 期 、 型 号 、 图 片 、 库 存 、 销 售 量 、 商 品类 型 、 会 员 
价 、 市 场 价 、 是 否 打折 等 属性 。 商 品 信息 实体 E-R 图 如 图 26.8 所 示 。 

除 上 面 介绍 的 两 个 E-R 图 ， 还 有 商品 订单 实体 、 商 品评 价 实体 、 公 告 实体 、 管 理 员 实 体 和 类 型 实体 和 
友情 链接 实体 等 ， 限 于 篇 幅 ， 这 里 仅 列 出 主要 的 实体 E-R 图 。 


图 26.7 “会员 信息 实体 E-R 图 26.8 商品 信息 实体 E-R 图 
26.3.2 ”创建 数据 库 和 数据 表 


区 1 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 创 建 数据 库 和 数据 表 .exe 

系统 E-R 图 设计 完成 后 ， 接 下 来 根据 E-R 图 来 创建 数据 库 和 数据 表 。 首 先 来 看 一 下 电子 商务 平台 所 使 
用 的 数据 表情 况 ， 如 图 26.9 所 示 。 

下 面 来 看 各 个 数据 表 的 结构 和 字段 说 明 。 

tb_admin (管理 员 信息 表 ) 

管理 员 信息 表 主 要 用 于 存储 管理 员 的 信息 ， 其 结构 如 图 26.10 所 示 。 


表 习作 记 好 匆 轨 。 类 型 可 可 
也 -admin 咒 辐 了 国 X 1 SAM utE_unicnde 
rass 大 时 加 让 国 X 5 MISAY We_unleore_ 
也 commo 喇 海 加 素面 XX 6 NISAM ule unisose_: 
了 am 咒 宣 加 蒜 夺 XX 4 MGA ulicode_e 
wke 硬 宙 加 天国 XS ence 2 
也 opinion 方略 守 定 XX 0 MYSAM ut unisode 5 auto_ineremert 
也 pubic 陨 呈 El 国 X 5 MyiSAM ae_unicode 
也 wer 喇 玉 网 肝 硬 XX 3 MSA ue_unleoue 
图 26.9 电子 商务 数据 表 图 26.10 管理 员 信息 表 结 构 


回 tb_class〈 商 品类 型 列表 ) 
商品 类 型 列表 主要 用 于 添加 商品 的 类 别 ， 可 以 设 定 多 个 子 类 别 〈 目 前 最 多 只 能 到 二 级 子 类 别 ) ， 其 结 


@ 
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构 如 图 26.11 所 示 。 
tb_commo (商品 信息 表 ) 
商品 信息 表 主 要 用 于 存储 关于 商品 的 相关 信息 ， 其 结构 如 图 26.12 所 示 。 


nth) 


hart20) ute_unieoce_ rl 


一 避 副 码 动 串 动 到 动 坊 动 串 码 吧 动 翅 夺 


图 26.11 商品 类 型 列表 结构 图 26.12 商品 信息 表 结 构 
此 外 还 有 商品 订单 表 、 商 品 公告 表 、 用 户 信息 表 、 友 情 链接 表 和 商品 留言 表 ， 限 于 篇 幅 ， 这 里 不 再 介 
绍 ， 读 者 可 参见 本 书 附 赠 光 盘 中 的 数据 库 文件 。 


26.4 公共 文件 设计 


多 4 视频 讲解 :光盘 \TM\Video\ 第 26 章 \ 公 共 文件 设计 .exe 

公共 模块 就 是 将 多 个 页 面 都 可 能 使 用 到 的 代码 写成 单独 的 文件 , 在 使 用 时 只 要 用 include 或 require 语句 
将 文件 包含 进来 即 可 。 如 本 系统 中 的 数据 库 连 接 、 管 理 和 分 页 类 文件 ，Smarty 模板 配置 类 文件 ， 类 的 实例 
化 文件 ，CSS 样式 表 文 件 , js 脚本 文件 等 。 以 前 台 系 统 为 例 ， 下 面 给 出 主要 的 公共 文件 ， 后 台 的 公共 文件 与 
前 台大 同 小 异 。 


26.4.1 数据 库 连 接 、 管 理 和 分 页 类 文件 


在 数据 库 连 接 、 管 理 和 分 页 类 文件 中 ， 定 义 3 个 类 ， 分 别 是 ConnDB 数据 库 连 接 类 ， 实 现 通过 PDO 连 
接 MySQL 数据 库 ，AdminDB 数据 库 管 理 类 ， 使 用 PDO 类 库 中 的 方法 执行 对 数据 库 中 数据 的 查询 、 添 加 、 
更 新 和 删除 操作 ; SepPage 分 页 类 ， 用 于 对 商城 中 的 数据 进行 分 页 输出 。 
(代码 位 置 ， 光盘 \TM\InstanceW26\system\system.class.inc.php) 
<?php 
/| 数据库 连接 类 
class ConnDB{ 
var $dbtype; 
var $host; 
var $user; 
var $pwd; 
var $dbname; 
function ConnDB($dbtype, $host, $user,$pwd, $dbnameX{ /构造 方法 
Sthis->dbtype=$dbtype; 
Sthis->host=$host; 
S$this->user=$Uuser; 
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Sthis->pwd=$pwd; 
S$this->dbname=$dbname; 


} 
function GetConnld(){ 
if($this->dbtype=="mysql" || $this->dbtype=="mssql"X{ 


// 实 现 数据 库 的 连接 并 返回 连接 对 象 


S$dsn="S$this->dbtype:host=$this->host;dbname=$this->dbname"; 


}else{ 


. 
try{ 


$dsn="$this->dbtype:dbname=$this->dbname"; 


$conn = new PDO($dsn, $this->user, $this->pwd); // 初 始 化 PDO 对 象 ， 就 是 创建 了 数据 库 连 接 对 象 $pdo 


S$conn->query("set names utf8"); 


return $conn; 
} catch (PDOException $e) { 
die ("Error!: " . $e->getMessage() . "<br/>"); 
} 
上 

} 

// 数 据 库 管理 类 

class AdminDB{ 


function ExecSQL($sqlstr, $connX{ 
$sqltype=strtolower(substr(trim($sqlstr),0,6)); 
S$rs=$conn->prepare($sqlstr); 
$rs->execute(); 
if($sqltype=="select"X{ 
$array=$rs->fetchAll(PDO::FETCH_ASSOC); 
if(count($array)==0 || $rs==false) 
return false; 
else 
return $array; 


// 准 备查 询 语句 
// 执 行 查询 语句 ， 并 返回 结果 集 


/获取 结果 集中 的 所 有 数据 


}elseif ($sqltype=="update" || $sqltype=="insert" || $sqltype=="delete")f 


if($rs) 

return true; 
else 

return false; 


} 


} 
// 分 页 类 
class SepPage{ 


Var $rs; 
var $pagesize; 
var $nowpage; 
var $array; 
var $conn; 
var $sqlstr; 
function ShowData($sqlstr, $conn,$pagesize,$nowpage\{ 
if(!isset($nowpage) || $nowpage=="") 
Sthis->nowpage=1; 
else 
Sthis->nowpage=$nowpage; 
Sthis->pagesize=$pagesize; 
S$this->conn=$conn; 
Sthis->sqlstr=$sqlstr; 


/定义 方法 
// 潮 断 变量 值 是 否 为 空 
/定义 每 页 起 始 页 


/定义 每 页 输出 的 记录 数 
/连接 数据 库 返 回 的 标识 
// 执 行 的 查询 语句 
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S$offset=($this->nowpage-1)*$this->pagesize; 

$sql=$this->sqlstr." limit $offset, $this->pagesize"; 

Sresult=$this->conn->prepare($sql); /准备 查询 语句 
Sresult->execute(); /| 执行 查询 语句 ， 并 返回 结果 集 
S$this->array=$result->fetchAll(PDO::FETCH_ASSOC):  // 获 取 结 果 集中 的 所 有 数据 
if(count($this->array)==0 || $this->array==false) 


return false; 
else 
return $this->array; 
} 
function ShowPage($contentname,$utits,$anothersearchstr,$anothersearchstrs,$classjf 
$str= 
Sres=$this->conn->prepare($this->sqlstr); /准备 查询 语句 
$res->execute(); /执行 查询 语句 ， 并 返回 结果 集 
Sthis->array=$res->fetchAlll(PDO::FETCH_ASSOC); /获取 结果 集中 的 所 有 数据 
$record=count($this->array); /| 统计 记录 总 数 
$pagecount=ceil($record/$this->pagesize); /计算 共有 几 页 


S$str.=$contentname."&nbsp;".$record."&nbsp;".$utits."&nbsp; 每 页 &nbsp;".$this->pagesize."&nbsp; 
".$utits."&nbsp; 第 &nbsp;".$this->nowpage."&nbsp; 页 / 共 &nbsp:;".$pagecount."&nbsp; 页 "; 
Sstr.="&nbsp;&nbsp;&nbsp;&nbsp;"; 
if($this->nowpage!=1) 

$str.="<a 
href=".$_SERVER[PHP_SELF'."?page=1&page_type=".$anothersearchstr."&parameter2=".$anothersearchstr 
s." Class=".$class."> 首 页 </a>"; 

else 

$str.="<font color=#555555'> 首 页 </font>"; 
Sstr.="&nbsp;"; 
if($this->nowpage!=1) 

Sstr.="<a 
href=".$_SERVER[PHP_SELF']."?page=".($this->nowpage-1)."&page_type=".$anothersearchstr."&parameter2 
=".$anothersearchstrs." class=".$class."> 上 一 页 </a>"; 

else 

S$str.="<font color=#555555'> 上 一 页 </font>"; 

$str.="&nbsp:"; 
if($this->nowpage!=$pagecount) 

$str.="<a 
href=".$_SERVER[PHP_SELF'."?page=".($this->nowpage+1)."&page_type=".$anothersearchstr."&parameter 
2=".$anothersearchstrs." class=".$class."> 下 一 页 </a>"; 

else 

$str.="<font color=#555555'> 下 一 页 </font>"; 

$str.="&nbsp;"; 
if($this->nowpage!=$pagecount) 

$str.="<a 
href=".$_SERVER[PHP_SELF]."?page=".$pagecount."&page_type=".$anothersearchstr."&parameter2=".$ano 
thersearchstrs." class=".$class."> 尾 页 </a>"; 

else 

$str.="<font color=#555555'> 尾 页 </font>"; 

if(count($this->array)==0 || $this->array==false) 
return "无 数据 ! "; 

else 
return $str; 
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26.4.2 Smarty 模板 配置 类 文件 


在 Smarty 模板 配置 类 文件 中 配置 Smarty 模板 文件 、 编 译文 件 、 配 置 文件 等 文件 路 径 。 
(代码 位 置 ， 光盘 \TMVInstance\26\system\system.smarty.inc-php) 


<?php 
require("libs/Smarty/Smarty.class.php"); /包含 模板 文件 
class SmartyProject extends Smarty{ /定义 类 ， 继 承 模板 类 
function SmartyProject()f /定义 方法 
Sthis->template_dir = "./system/templates/"; /指定 模板 文件 存储 位 置 
$this->compile_dir = "./system/templates_c/"; /指定 编译 文件 存储 位 置 
$this->config_dir = "./system/configs/"; /指定 配置 文件 存储 位 置 
S$this->cache_dir = "./system/cache/"; /指定 缓存 文件 存储 位 置 
} 
} 
?> 


26.4.3 ”执行 类 的 实例 化 文件 


在 system.inc.php 文件 中 , 通过 require 语句 包含 system.smarty.inc.php 和 system.class.inc.php 文件 , 执行 类 
的 实例 化 操作 ， 并 定义 返回 对 象 。 完 成 数据 库 连 接 类 的 实例 化 后 ， 调 用 其 中 的 GetConnId0 方 法 连接 数据 库 。 
(代码 位 置 ， 光盘 \TM\Instance\26\system\system.inc.php) 


<?php 
require("system.smarty.inc.php"); // 包 含 Smarty 配置 类 
require("system.class.inc.php"); // 包 含 数据 库 连 接 和 操作 类 
$connobj=new ConnDB("mysql","localhost","root","111","db_business"); /| 数据 库 连 接 类 实例 化 
$conn=$connobj->GetConnld(); /执行 连接 操作 ， 返 回 连 接 标识 
$admindb=new AdminDB(); /| 数据 库 操作 类 实例 化 
$seppage=new SepPage(); // 分 页 类 实例 化 
Susefun=new UseFun(); /使 用 常用 函数 类 实例 化 
$smarty=new SmartyProject(); // 调 用 Smarty 模板 
function unhtml($paramsi{ 

extract($params); 

$text=$content; 


global $usefun; 
return $usefun->UnHtml($text); 
$smarty->register_function("unhtml","unhtml"); // 注 册 模 板 函 数 
?> 
26.5 前 台 首页 设计 
医 1 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 前 台 首页 设计 .exe 
26.5.1 前 台 首页 概述 


前 台 首页 一 般 没 有 多 少 实质 的 技术 ， 主 要 是 加 载 一 些 功能 模块 ， 如 登录 模块 、 导 航 栏 模块 、 公 告 栏 模 
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块 等 ， 使 浏览 者 能 够 了 解 网 站 内 容 和 特点 。 首 页 的 重要 之 处 是 要 合理 地 对 页 面 进行 布局 ， 既 要 尽 可 能 地 将 
点 模块 显示 出 来 ， 同 时 又 不 能 因为 页 面 凌乱 无 序 ， 而 让 浏览 者 无 所 适 从 、 产 生 反感 。 本 系统 前 台 首页 的 
运行 结果 如 图 26.13 所 示 。 


9 亲征 两 征 a 


a ee 
26.5.2 ”Smarty 模板 页 中 的 框架 技术 


在 前 台 首 页 中 应 用 Switch 语句 与 Smarty 模板 中 的 内 建 函数 include 设计 一 个 框架 页 面 ， 实 现 不 同 功能 
模块 在 首页 中 的 展示 。 

Switch 语句 在 PHP 动态 文件 中 使 用 ， 根 据 超 链接 传递 的 值 ， 包 含 不 同 的 功能 模块 。 

Include 标签 在 Smarty 模板 页 中 使 用 ， 在 当前 模板 页 中 包含 其 他 模板 文件 。 其 语法 如 下 : 

{include file="file_name " assign=" " var="” "} 

file 指定 包含 模板 文件 的 名 称 ; assign 指定 一 个 变量 保存 包含 模板 的 输出 ; var 传递 给 待 包 含 模板 的 本 地 
参数 ， 只 在 待 包含 模板 中 有 效 。 


26.5.3 前台 首 页 实现 过 程 


(1) 创 建 ndex.php 动态 页 .在 index.php 动态 页 中 ,应 用 include_once0 语 句 包含 相应 的 文件 ,应 用 switch 
语句 ， 以 超 链接 中 参数 page 传递 的 值 为 条 件 进 行 判断 ， 实 现在 不 同 页 面 之 间 的 跳 转 。index.php 的 关键 代码 
如 下 : 

(代码 位 置 ， 光盘 \TMNVInstance\26\index.php) 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 


ifisset($_GET["page"]) 
$page=$_GET[T"page"]; 
Jelse{ 
$page=”'; 
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3 
include_once("login.php"); 
include_once("public.php"); 
include_once("links.php"); 
switch($page){ 
case "hyzx": 
include_once "member.php"; 
$smarty->assign(admin_phtml',,membertpl); /将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 
case 'allpub': 
include_once 'allpub.php'; 
$smarty->assign('admin_phtml",'allpub.tpl); /将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 
case 'nom': 
include_once 'allInom.php'; 
$smarty->assign('admin_phtml",'allInom.tpI); /将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 
case 'new': 
include_once 'allInew.php'; 
$smarty->assign('admin_phtml','aliInew.tpl); /将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 
case "hot : 
include_once 'allhot.php'; 
$smarty->assign(admin_phtml',allhottpl); /将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 
case 'shopcar': 
include_once 'myshopcar.php'; 
$smarty->assign(admin_phtml',myshopcartpl); // 将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 
case 'settle': 
include_once 'settle.php'; 
$smarty->assign(admin_phtml',settle tp); /将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 
case 'queryform': 
include_once 'queryform.php'; 
$smarty->assign('admin_phtml','queryform.tpl);// 将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 
default: 
include_once 'newhot.php'; 
$smarty->assign(admin_phtml',newhottpl); /将 PHP 脚本 文件 对 应 的 模板 文件 名 称 赋 给 模板 变量 
break; 


上 
$smarty->display("index.tpl"); /指定 模板 页 
?> 
(2) 创建 ndex.tpl 模板 页 。 在 模板 文件 index.tpl 中 应 用 Smarty 的 include 标签 调用 不 同 的 模板 文件 ， 
生成 静态 页 面 。 其 关键 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\26\system\templates\index.tp1) 
<table width="850" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td colspan="2">{include file='top.tpl}</td> 
</tr> 
<tr> 
<td width="216" align="left" valign="top"> 
{include file='login.tpl} 
{include file="public.tpl} 
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{include file="links.tpl} 
</td> 
<td width="634" height="700" align="center" valign="top"> 
{include file='search.tpl} 
<!-- 载 入 模板 文件 -->{include file=$admin_phtml}</td> 
</tr> 
</table> 
<table width="850" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td>{include file='buttom.tpl}</td> 
</tr> 
</table> 


py 
抽 明 本 系统 的 功能 较 多 ， 结 构 比较 复杂 ， 对 于 初学 者 来 说 学 起 来 可 能 会 比较 困难 。 所以， 本 书 将 系 
统 中 的 各 个 功能 模块 所 涉及 的 文件 (如 PHP、TPL、CSS、JS 等 ) 尽 可 能 都 单独 实现 。 读 者 在 学 习 其 中 
某 个 模块 时 ， 可 以 将 相关 的 文件 统一 放 到 同一 个 目录 下 单独 测试 。 


26.6 登录 模块 设计 
名 il 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 登 录 模 块 设计 .exe 


26.6.1 登录 模块 概述 


用 户 登 录 模 块 是 会 员 功能 的 窗口 。 匿 名 用 户 虽然 也 可 以 访问 本 网 站 ， 
但 只 能 进行 浏览 、 查 询 等 简单 操作 ， 而 会 员 则 可 以 购买 商品 ， 并 且 能 享受 
超 低 价格 。 登 录 模 块 包括 用 户 注 册 、 用 户 登录 和 找 回 密码 3 部 分 ， 其 运行 
结果 如 图 26.14 所 示 。 st 


26.6.2 Ajax 无 刷新 验证 技术 国人 


(1) Ajax 技术 无 刷新 验证 用 户 名 是 否 被 占用 。 其 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\check.js) 
”form 为 传 入 的 表单 名 称 ， 本 上段 代码 为 register 表单 */ 
function chkname(formX{ 
/* ”如 果 name 文本 域 的 信息 为 空 ， 名 为 name1 的 div 标签 显示 如 下 信息 */ 
if(form.name.value=="", 
name1.innerHMRL="<font color=#FF0000> 请 输入 用 户 名 ! </font>"; 
jelsef 
上 * 和 理 则 获取 文本 域 的 值 */ 
var user = form.name.value; 
广 生成 url 链接， 将 user 的 值 传 到 chkname.php 页 进行 判断 */ 
var url = "chkname.php?user="+user; 
A* ”使 用 xmlhttprequest 技术 运行 页 面 */ 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4X{ 
”根据 不 同 的 返回 值 ， 在 div 标签 中 输出 不 同 信息 */ 
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var msg = xmlhttp.responseText; 
if(msg == 3 
name1.innerHMRL="<font color=#FF0000> 用 户 名 被 占用 ! </font>"; 
return false; 
jelse if(msg == "2°X{ 
name1.innerHMRL="<font color=green> 葵 喜 您 ， 可 以 注册 !</font>"; 
六 如 果 用 户 名 正确 ， 则 将 隐藏 域 的 值 改 为 yes */ 
form.c_name.value = "yes"; 
}else{ 
name1.innerHMRL="<font color=green> 未 知 错误 </font>"; 
} 


} 
es 
} 
在 该 函数 中 调用 chkname.php 页 ， 该 页 在 会 员 登 录 时 也 会 被 调用 ， 所 以 这 里 分 两 种 情况 ， 有 密码 和 无 密 
码 。 无 密码 为 注册 验证 ， 当 没有 返回 结果 时 ， 说 明 该 用 户 名 可 用 ; 而 有 密码 为 登录 验证 ， 和 无 密码 相反 ， 
只 有 查询 记录 存在 时 , 才 人 允许 登录 ,并 将 用 户 名 和 用 户 ID 存储 到 session 中 。chkname.php 页 面 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\chkname.php) 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 
S$reback = '0'; 


$sql = "select * from tb_user where name='".$_GET[user].""; 
if(isset($_GET['password'])\{ 
$sql .=" and password = ".md5($_GET[password]).""; 


} 
Srst = $admindb->ExecSQL($sql,$conn); 
if($rst}X{ 
/登录 */ 
if($rst[O][isfreeze'] != OX{ 
Sreback = '3'; 
}else{ 
$_SESSION[member] = $rst[0l[name']; 
$_SESSION['id] = $rst[olrid]; 
S$reback = '2'; 


} 
Jelse{ 
S$reback = "1"; 


echo S$reback; 

?> 
(2) GD2 函数 库 生 成 验证 码 。 其 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance26\yzm.php) 


<?php 

header ( "Content-type: text/html; charset=UTF-8" ); /1 设置 文件 编码 格式 
srand((double)microtime()*1000000); /生成 随机 数 
$im=imagecreate(60,30); 1/ 创建 画布 
$black=imagecolorallocate($im,0,0,0); /定义 背景 
S$white=imagecolorallocate($im,255,255,255); /定义 背景 
$gray=imagecolorallocate($im,200,200,200): /定义 背景 
imagefill($im,0,0,$gray); // 填 充 颜色 
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for($i=0;$i<4;$i++)f /定义 4 位 随机 数 
$str=mt_rand(3,20); /定义 随机 字符 所 在 位 置 的 Y 坐标 
Ssize=mt_rand(5,8); /定义 随机 字符 的 字体 
$authnum=substr($_GET[num']$i1); /获取 超 链接 中 传递 的 验证 码 
imagestring($im, $size,(2+$i*15),$str,$authnum,imagecolorallocate($im,rand(0,130),rand(0,130),rand(0,130))); 
} /水 乎 输出 字符 串 
for($i=0;$i<200:;$i++) /执行 for 循环 ， 为 验证 码 添加 模糊 背景 
S$randcolor=imagecolorallocate($im,rand(0,255),rand(0,255),rand(0,255)); ” // 创 建 背景 
imagesetpixel($im,rand()%70,rand()%30,$randcolor); /绘制 单一 元 素 
1 
imagepng($im); /| 生成 PNG 图 像 
imagedestroy($im); // 销 毁 图 像 
?> 


26.6.3 用 户 注册 


用 户 注册 页 面 的 主要 功能 是 新 用 户 注 册 。 如 果 信息 输入 完整 而 且 符合 要 求 ， 则 系统 会 将 该 用 户 信息 保 
存 到 数据 库 中 ， 和 否则 显示 错误 原因 ， 以 便 用 户 改正 。 用 户 注册 页 面 的 运行 结果 如 图 26.15 所 示 。 


二 总 的 为 作 搞 唤 


图 26.15 注册 模块 页 面 
(1) 创建 registertpl 模板 文件 ， 编 写 用 户 注册 页 面 。 其 中 包含 两 个 JS 脚本 文件 createxmlhttpjs 和 check 


js。 其 中 ，createxmlhttp.js 是 Ajax 的 实例 化 文件 ,而 checkjs 对 用 户 注册 信息 进行 验证 ,并 且 返 回 验证 结果 。 


(2) 创建 register.php 动态 PHP 文件 ， 加 载 模板 。register.php 文件 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\26\register.php) 


<?php 
header ( "Content-type: text/html; charset=UTF-8" ); 1/ 设置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 


$smarty->assign('title',' 新 用 户 注册 '); 
$smarty->display('register.tp['); 
> 
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(3) 创建 reg_chk.php 文件 ， 获 取 表 单 中 提交 的 数据 ， 将 数据 存储 到 指定 的 数据 表 中 。reg_chkphp 文 
件 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\register.php) 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 


S$name = $_POST[name'; 

S$password = md5($_POST[pwd1"); 

Squestion = $_POST[question']; 

Sanswer = $_POST[answer]; 

Srealname = $_POST[realname']; 

S$card =$_POST['card]; 

S$tel = $_POSTItel]; 

S$phone = $_POST[phone'; 

$Email = $_POST[email]; 

$QQ = $_POST[qq]: 

$code = $_POST[code'; 

Saddress = $_POST[address]; 

$addtime = date("Y-m-d H:i:s"); 

$sql="insertinto 
tb_user(name,password,question,answer,realname,card,tel,phone,Email, QQ,code,address,addtime ,isfreeze,sh 
opping,consume)" ; 

$sql .= " values (‘$name', '$password', '$question', '$answer', '$realname', '$card", '$tel', '$phone', '$Email', 
'$QQ', '$code', '$address','$addtime','0',",'00.00")"; 

Srst= $admindb->ExecSQL($sql,$conn); 

if($rstX{ 
$_SESSION[member] = $name; 

echo "<script>top.opener.location.reload();alert( 注 册 成 功 ');window.close();</script>"; 
}else{ 

echo '<script>alert(\' 添 加 失败 \');history.back;</script>"; 
} 


?> 
(4) 创建 “用 户 注册 ” 超 链接 。 当 用 户 单 击 前 台 的 本 ED 丽 时 ， 系 统 会 调用 JS 文件 的 onclick 事件 ， 弹 
出 注册 窗口 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TMNInstance\26\system\vtemplates\login.tpl) 
<a href="#" id="login" onclick="reg()"><img src="images/check.JPG" width="59" height="23" border="0" /></a> 
这 里 使 用 到 的 JS 文件 为 js/loginjs， 调 用 的 函数 为 reg0。 该 函数 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\26\js\login.js) 
function reg(X{ 


window.open("register.php", "_blank", "width=600,height=650",false); // 弹 出 窗口 
区 
26.6.4 用 户 登 录 


用 户 登 录 模 块 的 运行 结果 如 图 26.16 所 示 ， 需 要 输入 用 户 名 、 密 码 和 验 
证 码 。 
(1) 创建 模板 文件 login.tpl， 完 成 用 户 登 录 表 单 的 设计 。 在 该 页 面 中 当 
单 击 Submit 按钮 时 , 系统 将 调用 lg0 函 数 对 用 户 登录 提交 信息 进行 验证 .lg0 
函数 包含 在 js/loginjs 脚本 文件 内 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\login.js) 图 26.16 用 户 登 录 页 面 


局 
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l/JavaScript Document 
function lg(formX{ 
if(form.name.value=="™"){ 
alert(' 请 输入 用 户 名 '); 


form.name.focus(); 


return false; 
} 
if(form.password.value == "" || form.password.value.length < 6X{ 
alert(' 请 输入 正确 密码 '); 
form.password.focus(); 
return false; 


} 

if(form.check.value == "}{ 
alert(' 请 输入 验证 码 '); 
form.check.focus(); 
return false; 


} 
if(form.check.value != form.check2.value}{ 
form.check.select(); 
code(form); 
return false; 
} 
var user = form.name.value; 
var password = form.password.value; 
var url = "chkname.php?user="+user+"&password="+password; 
xmlhttp.open("GET",urltrue); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
alert( 用 户 名 或 密码 错误 册 ); 
form.password.select(); 


form.check.value = "; 
code(form); 
return false; 

Hf(msg == "3 
alert(" 该 用 户 被 冻结 ， 请 联系 管理 员 "); 
return false; 

}else{ 
alert(' 欢 迎 光临 "); 
location.reload(); 

) 

; 
} 
xmlhttp.send(null); 
return false; 
} 
// 显 示 验 证 码 
function yzm(formX{ 


var num1=Math.round(Math.random()*10000000); 
var num=num1.toString().substr(0,4); 


document.write("<img name=codeimg width=65 heigh=35 src='yzm.php?num="+nuM+">"); 


form.check2.value=num; 


// 刷 新 验证 码 

function code(form){ 
var num1=Math.round(Math.random()*10000000); 
var num=num1.toString().substr(0,4); 


} 
// 注 
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document.codeimg.src="yzm.php?num="+num: 
form.check2.value=num; 


册 


function reg(){ 
window.open("register.php", "_blank", "width=600,height=650",false); 


} 
// 找 


回 密码 


function found() { 


Win 


dow.open("found.php","_blank","width=350 height=240",false); 


} 
用 户 名 和 密码 是 在 chkname.php 页 面 中 被 验证 的 。 chkname.php 在 26.6.2 节 中 已 经 介绍 过 , 这 里 不 再 重复 了 。 


(2 


) 创建 用 户 信息 模板 文件 info.tpl。 用 户 登 录 成 功 后 ， 在 原 登 录 框 位 置 将 显示 用 户 信 息 ， 用 户 可 以 通 


过 “会 员 中 心 ” 对 自己 的 信息 做 修改 ， 也 可 以 单 击 “ 查 看 购物 车 ” 超 链 接 查 看 购物 车 商品 ， 当 用 户 离开 时 


可 以 单 才 


Ff“ 安全 离开 ” 超 链 接 。 用 户 信息 模 块 的 主要 代码 如 下 : 


(代码 位 置 ， 光盘 \TM\Instance\26\system\templates\info.tp1) 


显示 当前 登录 用 户 名 --> 


<I-- 


欢迎 您 : {$member} 


< 上 


会 员 中 心 超 链接 一 > 


<a href="?page=hyzx" id="info" class="lk"> 会 员 中 心 </a> 


查看 购物 车 --> 


< 


<a href="?page=shopcar" class="lk"> 查 看 购物 车 </a> 


<|- 


安全 离开 -> 


<a onclick="javascript:logout()" style="cursor:hand" id="info"> 安 全 离开 </a> 


26.6.5” 找 回 密码 


登录 模块 的 最 后 一 个 部 分 就 是 找 回 密码 。 找 回 密码 是 根据 用 户 在 填写 资料 时 所 填写 的 密 保 问题 和 密 保 
答案 来 实现 的 。 当 用 户 单 击 “ 找 回 密码 ” 超 链接 时 ， 首 先 提示 用 户 输入 要 找 回 密码 的 会 员 名 称 ， 然 后 根据 
密 保 问 题 填写 密 保 答 案 ， 最 后 重新 输入 密码 。 找 回 密码 模块 的 流程 如 图 26. 17 所 示 。 


第 一 步 : 输入 会 员 名 称 


第 三 步 : 输入 新 密码 第 四 步 : 密码 修改 成 功 
图 26.17 找 回 密码 流程 图 
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1. 创建 模板 文件 
虽然 找 回 密码 需要 4 个 步骤 ， 但 实际 上 每 个 步骤 使 用 的 都 是 相同 的 模板 文件 和 JS 文件 ， 只 是 被 调用 的 
表单 和 js 函数 略 有 差别 。 这 里 根据 不 同 的 文件 来 分 别 进行 介绍 。 
该 模板 文件 一 共 包含 了 3 个 表单 ， 分 别 代表 了 3 个 步骤 。 其 核心 代码 如 下 : 
(代码 位 置 : 光盘 \TM\Instance\26\system\templates\found.tp1) 
<!-- 载 入 两 个 JS 脚本 文件 一 > 
<script language="javascript" src="js/createxmlhttp.js"></script> 
<script language="javascript" src="js/found.js"></script> 
<!-- 第 1 个 dv 标签 --> 
<div id="first"> 
<table width="200" border="0" cellspacing="0" cellpadding="0"> 
<form id="foundname" name="found" method="post" action="#"> 
<tr><td> 找 回 密码 </td></tr> 
<tr><td> 会 员 名 称 : </td> 
<!-- text 文本 域 ， 用 于 输入 要 找 回 密码 的 会 员 名称 ”一 > 
<td><input id="user' name="User" type="text" class="txt"></td> 
</tr> 
<tr><td> 
<!-- 单 击 “下 一 步 ” 按 钮 ， 能 触发 onclick 事件 来 调用 chkname 函数 ”--> 
<input id = " next1 " name = " next1 " type = " button " class = " btn " value = " 下 一 步 " onClick = " return 
chkname ( foundname ) "/></td></tr> 
</form> 
</table> 
</div> 
<!-- 第 2 个 div 标签 ， 样式 为 隐藏 --> 
<div id="second" style="display:none;"> 
<table> 
<form id="foundanswer" name="found" method="post" action="#"> 
<tr><td > 找 回 密码 </td></tr> 
<tr><td> 密 保 问题 </td> 
<!-- ”用 于 显示 密 保 问题 的 div 标签、 一 > 
<td <div id="question"></div></td></tr> 
<tr><td> 密 保 答案 : </td> 
<!-- 文本 域 ， 用 于 填写 密 保 答 案 --> 
<td ><input id="answer" name="answer" type="text" class="txt" /></td></tr> 
<tr> 
<!-- 单 击 “ 下 一 步 ”按钮 ， 用 来 触发 onclick 事件 ， 并 调用 chkanswer() 函 数 ”一 > 
<td><input id = " next2 " name = " next2 " type=" button " class=" btn " value =" 下 一 步 " onClick = " return 
chkanswer ( foundanswer ) "></td> 
</tr> 
</form> 
</table> 
</div> 
<!-- 第 3 个 div 标签 ， 样式 也 为 隐藏 ， 作 用 是 修改 密码 -> 
<div id='third' style="display:none;"> 
<table> 
<form id="modifypwd" name="found" method="post" action="#"> 
<tr><td> 输入 密码 </td></tr> 
<tr><td> 输 入 密码 : </td> 
<td><input id="pwd1" name="pwd1" type="password" class="txt"></td></tr> 
<tr><td> 确 认 密 码 : </td> 
<td><input id="pwd2" name="pwd2" type="password" class="txt" /></td> 
</tr> 
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<tr> 
<!- 单 击 “ 完 成 ”按钮 ， 调 用 ckpwd() 函 数 一 > 
<td><input id =" mod " name = “ mod " type = " button " class = " btn " value = " 完成 " onClick = " return 
chkpwd (modifypwd) "></td> 
</tr> 


</form> 
</table> 
</div> 
可 以 看 出 ， 在 上 述 3 个 表单 中 ， 只 有 一 个 表单 默认 情况 下 是 显示 的 ， 其 他 则 为 隐藏 。 只 有 通过 调用 不 
同 的 js 函数 ， 才 可 以 对 其 他 表单 进行 操作 。 
2. 创建 JS 脚本 文件 
foundjjs 脚本 文件 包含 3 个 函数 : chkname()、chkanswer0 和 chkpwd0。 其 中 ，chkname0 函 数 的 作用 是 
检查 用 户 输入 的 会 员 名 称 ， 如 果 存 在 ， 则 使 用 xmlhttp 对 象 去 调用 生成 的 url 进行 处 理 判断 。 如 果 该 用 户 存 
在 ， 则 隐藏 当前 表单 ， 并 显示 下 一 个 表单 ， 最 后 输出 密 保 问题 。chkname() 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\found.js) 
function chkname(formX{ 
var user = form.user.value; 
if(user == "X{ 
alert(' 请 输入 用 户 名 '); 
form.user.focus(); 
return false; 
}else{ 
var url = "foundpwd.php?user="+user; 
xmlhttp.open("GET",urltrue); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg =='0'X{ 
alert(' 没 有 该 用 户 ， 请 重新 查找 '); 
form.user.select(); 
return false; 
}else{ 
document.getElementByld(first).style.display = 'none’; 
document.getElementByld('second').style.display = "; 
document.getElementByld('question').innerHTML = msg; 


} 


} 
xmlhttp.send(null); 
} 


} 
其 他 两 个 函数 也 使 用 xmlhttprequest 对 象 ,实现 方法 相差 无 几 , 不 同 之 处 就 是 对 返回 值 的 处 理 ,chkanswer() 
函数 隐藏 当前 表单 ， 显 示 下 一 个 表单 。chkanswerO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TMNVInstance\26\jsfound.js) 
function chkanswer(form) { 
var user = document.getElementByld('user').value; 
var answer = form.answer.value; 
if(answer == "多 
alert( 请 输入 提示 问题 ); 
form.answer.focus(); 
return false; 


jelsef 
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var url = "foundpwd.php?user="+user+"&answer="+answer; 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4){ 
var msg = xmlhttp.responseText; 
if(msg == "0'X{ 
alert(' 问 题 回 答 错误 '); 
form.answer.select(); 
return false; 
}else{ 
document.getElementByld('second ').style.display = "none' 
document.getElementByld(third').style.display = "; 


} 


} 
xmlhttp.send(null); 
) } 
而 chkpwd0 函 数 则 提示 用 户 操作 状态 ， 如 果 成 功 ， 则 关闭 当前 页 。ckpwd0 函 数 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\26\js\found.js) 
function chkpwd(formX{ 
var user = document.getElementByld(user).value; 
var pwd1 = form.pwd1.value; 
var pwd2 = form.pwd2.value; 
if(pwd1 =="X{ 
alert(' 请 输入 密码 '); 
form.pwd1.focus(); 
return false; 


} 

if(pwd1.length < 6){ 
alert( 密码 输入 错误 ); 
form.pwd1.focus(); 
return false; 


} 

if(pwd1 != pwd2X{ 
alert(' 两 次 密码 不 相等 '); 
form.pwd2.select(); 
return false; 


var url = "foundpwd.php?user="+user+"&password="+pwd1; 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1 
alert(' 密 码 修改 成 功 ， 请 重新 登录 '); 
window.close(); 
jelse{ 
alert(msg); 
} 


| 


} 
xmlhttp.send(null); 


S93 


PHP+MySQL 开发 实战 


3. 创建 数据 处 理 文件 
foundpwd.php 文件 的 功能 是 根据 用 户 输 入 信息 来 检测 数据 表 中 的 数据 ， 并 根据 不 同 的 输入 信息 返回 不 同 
的 结果 。 该 文件 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance26\foundpwd.php) 


<?php 

header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 $smarty->assign('title',' 找 回 密码 '); 
S$reback = '0'; // 设 置 变量 初始 值 


iflisset($_GET[answer]) && lisset($_GET[password"])}{ // 判 断 变 量 是否 存 在 
$namesql = "select * from tb_user where name = ".$_GET[user].""; 
$namerst = $admindb->ExecSQL($namesql,$conn); /查询 用 户 名 是 否 存在 
if($namerstX{ 
$question = $namerst[0][question]; 
Sreback = $question; 


} 
}else iflisset($_GET[answer])){f 
$answersql = "select * from tb_user where name = ".$_GET[user]." and answer = ".$_GET[answer].""; 
$answerrst = $admindb->ExecSQL($answersql,$conn); 
if($answerrst}{ 
Sreback = "1'; 


} 

}else if(isset($_GET['password'])X{ 
$sql="update tb_user set password=".md5($_GET[password'"])." where name=".$_GET[user].""; 
Srst = $admindb->ExecSQL($sql,$conn); 


if($rstX{ 
Sreback = "1'; /为 模板 变量 赋值 
1 
echo S$reback; /输出 返回 结果 
Ye 
4. 加 载 模板 页 


因为 所 有 登录 模块 的 模板 都 不 需要 或 者 只 需要 传递 一 两 个 变量 ， 所 以 PHP 加 载 页 的 内 容 比较 简单 。 找 
回 密码 页 面 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\found.php) 
<?php 


header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配 置 文 件 $smarty->assign('title',' 找 回 密码 '); 
$smarty->display('found.tp!"); 

?> 


26.7 会 员 信息 模块 设计 
吉 4 视频 讲解 光盘 \TM\Video\ 第 26 章 \ 会 员 信息 模块 设计 .exe 


26.7.1 会 员 信息 模块 概述 


用 户 登 录 后 ， 即 可 看 到 会 员 信息 模块 。 在 这 里 ， 可 以 进行 查看 或 修改 个 人 信息 及 密码 、 查 看 购物 车 和 


@ 
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安全 退出 等 操作 。 本 节 只 对 会 员 信息 模块 中 的 “会 员 中 心 ” 和 “安全 退出 ”进行 讲解 ， 关 于 “查看 购物 车 ” 
将 在 商品 模块 中 进行 介绍 。 会 员 信息 模块 的 运行 效果 如 图 26.18 所 示 。 


En 
RE 


闻 ; 区 DIE05 了 
二 
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图 26.18 会 员 中 心 
26.7.2 会 员 信 息 查 询 技术 


在 会 员 信息 模块 中 ， 以 SESSION 变量 中 存储 的 用 户 名 称 为 条 件 ， 从 会 员 信息 表 中 查询 出 会 员 信息 ， 并 
且 将 会 员 信息 存储 到 模板 变量 中 ， 最 后 在 模板 页 中 输出 会 员 信息 。memberphp 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\member.php) 
<?php 
/* ”查找 用 户 资料 */ 
iflisset($_SESSION[member])X{ 
$sql = "select * from tb_user where name = ".$_SESSION['member]."™; 
$arr = $admindb->ExecSQL($sql,$conn); 
ifisset($_GET[action]) && $_GET[action] == ,modify)f 


$smarty->assign(check',"find"); 
$smarty->assign('pwdarr', $arr); 
}else{ 
$smarty->assign('check',"notfind"); 
$smarty->assign(pwdarr,$arr); 
} 
3 
?> 


memberphp 文件 中 查询 出 的 数据 是 会 员 信息 模板 功能 实现 的 根本 。 
26.7.3 会员 中 心 


当 单 击 “ 会 员 中 心 ” 超 链接 时 , 会 回 传 给 当前 页 一 个 page 值 , 当前 页 根据 这 个 page 值 来 载 入 member.php 
文件 。 

1. 创建 PHP 页 面 

与 登录 模块 设计 不 同 ， 本 节 首 先 来 创建 PHP 页 面 。 因 为 该 模块 中 的 模板 需要 使 用 数据 库 中 的 数据 及 一 
些 动态 信息 ， 这 些 都 需要 在 PHP 页 中 先行 获取 及 处 理 ， 然 后 再 传 给 模板 页 。 会 员 中 心 页 面 的 代码 请 参考 技 


术 分 析 中 的 内 容 。 
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2. 创建 模板 页 


该 模块 包括 查看 信息 模板 及 修改 密码 模板 ， 都 存储 于 member.tpl 模板 文件 中 。 
(代码 位 置 ， 光盘 \TM\Instance\26\system\templates\member.tp1) 
<link rel="stylesheet" href="css/member.css" /> 
<script language="javascript" src="js/member.js"></script> 
{if $check=="find" } 
<p align="left">{$smarty.session.member}&gt;&gt;&gt;<a href='?page=hyzx' id="mem"> 查 看 信息 </a>&gt;&gt; 
&gt;<a href='?page=hyzx&action=modify id="mem"> 修 改 密码 </a></p> 
<table id="member" width="300" border="0" cellpadding="0" cellspacing="0"> 
<form id="member" name="member” method="post" action="modify_pwd_chk.php" onSubmit="return 
pwd(member)"> 
<tr> 
<td height="25" colspan="2”align="center”valign="middle”id="first"><font color="#f0f0f0"> 修改 密码 
</font></td> 
</tr> 
ee // 省 略 了 部 分 代码 
<tr> 
<td height="30" colspan="2" align="center" valign="middle"><input id="enter" name="enter" type="submit" 
value=" 修 改 " /></td> 
</tr> 
</form> 
</table> 
{else} 
<p align="left">{$smarty.session.member}&gt;&gt;&gt;<a href='?page=hyzx' id="mem"> 查 看 信息 </a>&gt;&gt; 
&gt;<a href='?page=hyzx&action=modify id="mem"> 修 改 密码 </a></p> 
{section name=pwd_id loop=$pwdarr} 
<table id='member width="500" border="0" cellpadding="0" cellspacing="0"> 
<form id="member"” name="member"” mer ="post" action="modify_info_chk.php” onSubmit="return 
mem(member)" > 


<tr> 
<td height="25" colspan="2" align="center" valign="middle" id="first"><font 
color="#f0f0f0">{$pwdarr[fpwd_id].name} 信 息 (不 可 更 改 信息 ) </font></td> 
</tr> 
<tr> 


<td width="25%" height="25" align="right" valign="middle" id="left"> 会 员 编号 : </td> 
<td height="25" align="left" valign="middle" id="right">&nbsp;{$pwdarr[pwd_id].id}</td> 
</tr> 
ck // 省 略 了 部 分 代码 
<tr> 
<td height="30" colspan="2" align="center" valign="middle"><input name="enter" type="submit" id="enter" 
value=" 修 改 " />&nbsp;&nbsp;&nbsp;&nbsp;<input name="reset" type="reset" id="reset" value=" 重 置 " /></td> 
</ltr> 
</form> 
</table> 
{/section} 
{i 
3. 创建 脚本 文件 
该 模块 的 脚本 文件 和 用 户 注册 模块 类 似 ， 都 是 对 信息 的 合法 性 进行 验证 ， 如 信息 是 否 为 空 、 是 否 符合 
规范 等 ， 这 里 就 不 再 袭 述 了 。 
4. 创建 处 理 页 
当 信息 验证 通过 后 ， 系 统 将 跳 转 到 处 理 页 进行 信息 处 理 。 本 模块 处 理 页 分 信息 修改 和 密码 修改 两 个 页 


他 
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面 。 首 先 介 绍 信息 修改 页 ， 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance26\modify_info_chk.php) 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 


$sql = "update tb_user set realname=".$_POST[realname'].",card=".$_POST[card'].",tel=".$_POST['tel].", 
phone= ".$_POST[phone].",Email=".$_POST[email].",QQ='".$_POST[qq].",code=".$_POST[code]."， 
address=".$_POST[address']." where id = ".$_POST[userid].”"; 
$arr = $admindb->ExecSQL($sql,$conn); 
if($arr) 
echo "<script>alert(' 修 改 成 功 ');location=('index.php');</script>"; 
else 
echo "<script>alert(' 修 改 失 败 ');history.go(-1);</script>"; 
了 
与 信息 修改 页 的 操作 流程 十 分 类 似 ， 只 是 更 新 的 数组 要 小 得 多 ， 只 有 一 个 字段 。 修 改 密码 页 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\modify_pwd_chk.php) 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 

$sql="select * from tb_user where name = ".$_SESSION[member]." and password=".md5($_POST[old)])." "; 
$arr = $admindb->ExecSQL($sql,$conn); // 削 断 用 户 名 和 密码 是 否 正确 

if($arrX{ 


$sql = "update tb_user set password=".md5($_POST[new1")." where name = ".$_SESSION['member]." 
and password=".md5($_POST[old)])." "; 

$arr = $admindb->ExecSQL($sql,$conn); 

echo "<script>alert(' 密 码 修改 成 功 ! '); window.location.href='index.php';</script>"; 


}else{ 
echo "<script>alert(' 密 码 修改 失败 ! '); window.location.href='index.php';</script>"; 
} 


?> 


26.7.4 安全 退出 


当 用 户 需要 离开 网 站 时 ， 可 以 单 击 “ 安 全 退出 ” 超 链接 来 调用 logout0 函 数 ， 当 用 户 确认 退出 后 ， 则 跳 
转 到 logout 页 面 ， 销 毁 session 并 回 到 首页 。 安 全 退出 所 涉及 的 页 面 及 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\26\js\info.js) 
function logout(){ 


访 confirm(" 确 定 要 退出 登录 吗 ? ")X{ /输出 选择 框 ， 用 户 可 以 单 击 “ 确 认 ” 或 “取消 ”按钮 
window.open('logout.php','_parent',",false); // 如 果 用 户 确认 退出 ， 则 打开 logout.php 页 

jelse 

return false; 


} 
(代码 位 置 ， 光 盘 \TM\Instance\26\logout.php) 

<?php 
session_start(); 
header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
session_destroy(); 
echo '<script>alert(\ 用 户 已 安全 退出 N\);location=(\'index.php\);</script>"; 

?> 


PHP+MySQL 开发 实战 


26.8 商品 展示 模块 设计 
苹 g 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 商 品 展示 模块 设计 .exe 
26.8.1 商品 展示 模块 概述 
本 系统 为 用 户 提供 了 不 同 的 商品 展示 方式 ， 包 括 推荐 商品 、 最 新 商品 、 热 门 商品 等 ， 能 够 使 消费 者 有 


目的 地 选 购 商品 。 每 个 展示 方式 中 都 有 商品 详细 信息 的 显示 ， 为 用 户 购买 商品 提供 可 靠 的 依据 。 本 系统 商 
品 显示 模块 的 运行 结果 如 图 26.19 所 示 。 


26.19 ”商品 展示 模块 页 面 


因为 推荐 商品 、 最 新 商品 和 热门 商品 的 实现 方法 和 过 程 基本 相同 ， 所 以 本 节 只 讲解 推荐 商品 模块 。 其 
他 功能 的 相关 代码 可 参见 光盘 中 的 源 程序 。 


26.8.2 ”分 页 技术 


商品 显示 功能 实现 的 关键 就 是 如 何 从 数据 库 中 读 取 商 品 信息 ， 如 何 完成 数据 的 分 页 显示 。 首 先 ， 包 含 
类 的 实例 化 文件 。 然 后 ， 判 断 分 页 变量 page 的 值 是 否 存在 。 接 着 ， 定 义 SQL 语句 ， 对 查询 结果 进行 降 暴 排 
列 ， 并 且 设 置 每 页 显示 3 条 记录 。 再 调用 分 页 类 中 的 方法 完成 数据 的 分 页 读 取 和 输出 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TMVInstance\26\allinom.php) 


<?php 
include_once("system/system.inc.php"); // 包 含 类 的 实例 化 文件 
iflisset($_GET[page]))f // 判 断 当前 页 变量 的 值 
$page=$_GET[page]; 
Jelse{ 
$page=1; 
$rst1 = $seppage->ShowData("select * from tb_commo where isnom = 1 order by isnom,id desc",$conn,3,$page); 
// 调 用 分 页 类 方法 
$smarty->assign("nomarr",$rst1); 
$smarty->assign('rst1_page',$seppage->ShowPage(" 产 品 "," 个 ",$_GET[page_type'],","a")); 
/| 输出 分 页 超 链接 


$smarty->assign('title', 推 荐 商品 "); 
?> 


局 
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最 后 ， 定 义 模板 文件 ， 通 过 section 语句 循环 输出 存储 在 模板 变量 中 的 数据 ， 并 且 输 出 分 页 超 链接 。 
26.8.3 ”商品 展示 模块 的 实现 过 程 


在 技术 分 析 中 已 经 对 商品 显示 所 使 用 的 技术 、 方 法 进行 概述 ， 下 面 就 介绍 一 下 它 具 体 的 过 程 。 
(1) 创建 allnom.php 文件 ， 从 数据 库 中 读 取出 推荐 商品 的 数据 ， 并 将 数据 存储 到 模板 变量 中 ， 其 代码 
可 以 参考 技术 分 析 。 
(2) 创建 allnom.tpl 模板 页 ， 应 用 section 语句 输出 商品 信息 ， 并 添加 相应 的 操作 按钮 或 链接 。 模 板 页 
中 一 共有 两 个 事件 : 查看 商品 详细 信息 和 放 入 购物 车 。 
当 单 击 “查看 详情 ”按钮 时 ， 将 触发 onclick 事件 ， 并 将 调用 openshowcommo0 函 数 ， 同 时 ， 商 品 
id 会 作为 函数 的 唯一 参数 被 传递 进去 。 
回 ” 当 单 击 “ 购 买 ” 按钮 时 ， 同 样 会 触发 onclick 事件 ， 并 调用 buycommo0 函 数 ， 唯 一 的 参数 也 是 商品 
的 id。 
商品 模板 页 面 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\system\templates\allnom.tpl) 
ee width="636" border="0" align="center" cellspacing="0" cellpadding="0"> 
i height="33" align="left" valign="middle" background="images/shop_07.gif’>&nbsp;</td> 
</tr> 


<tr> 
<td height="132" align="|left" valign="middle"> 
{section name=nom_id loop=$nomarr} 
<table width="636" height="134" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td width="145" rowspan="5" align="center"” valign="middle"><img src="{$nomarr[Inom_id].pics}" 
width="90" height="100" alt="{$nomarr[nom_id].name}" style="border: 1px solid #fOf0f0;" /></td> 
<td height="35"> 商 品名 称 : {$nomarr[nom_id].name}</td> 
<td width="156" height="35"> 商 品类 别 : {$nomarrInom_id].class}</td> 
<td width="157"> 商 品 型 号 : {$nomarr[nom_id].model}</td> 
</tr> 
<tr> 
<td height="23"> 商 品 品 牌 : {$nomarrInom_id].brand}</td> 
<td height="23" colspan="2"> 商 品 产地 : {$nomarrInom_id].area}</td> 
</tr> 
<tr> 
<td width="178" height="23"> 剩 余数 量 : {$nomarr[nom_id].stocks}</td> 
<td colspan="2"> 销 售 数量 : {$nomarrInom_id].sell}</td> 
</tr> 
<tr> 
<td height="23"> 市 场 价 : <font color="red">{$nomarrInom_id].m_price}&nbsp; 元 </font></td> 
<td height="23" colspan="2"> 上 市 日 期 : {$nomarr[Inom_id].addtime}</td> 
</tr> 
<tr> 
<td height="30"> 会 员 价格 : <font color="#FF0000">{$nomarrInom_id].v_price}&nbsp; 元 </font></td> 
<td height="30" colspan="2" align="center" valign="middle"><input id="allshow" name="allshow" 
type="button" value="" class="showinfo" onclick="openshowcommo({$nomarrInom_id].id})” /> &nbsp;<input 
id="buy" name="buy" type="button" value="" class="buy" onclick="return buycommo({$nomarrIinom_id].id})" 
/></td> 
</tr> 
</table> 
<hr style="border: 1px solid #0f0f0;" /> 
{/section} 


全 
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</td> 
</tr> 
<tr> 
<td height="25">{$rst1_page}</td> 
</tr> 
</table> 
(3) 创建 showcommojs 脚本 文件 。 当 单 击 “ 查 看 商品 ”按钮 时 ， 系 统 会 弹出 一 个 新 的 页 面 ， 并 显示 
商品 的 详细 信息 ， 当 单 击 “ 购 买 ”按钮 时 ， 该 商品 将 会 被 放 到 当前 用 户 的 购物 车 中 ， 如 果 没 有 登录 用 户 或 
商品 已 添加 ， 则 会 提示 错误 信息 。JS 脚本 文件 的 代码 如 下 : 
(代码 位 置 : 光盘 \TM\Instance\26\js\showcommo.js) 
/* ”查看 商品 信息 函数 ， 将 打开 一 个 新 页 面 。*/ 
function openshowcommo(key)\{ 
open(showcommo.php?id='+key，blank',' width=560 height=300',false); 


} 
/* 将 购买 商品 添加 到 购物 车 中 ， 将 在 下 节 中 讲解 */ 
function buycommo(key){ 


} 
26.9 ”购物 车 模块 设计 


茹 4 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 购 物 车 模块 设计 .exe 


26.9.1 ”购物 车 模块 概述 


购物 车 是 电子 商务 平台 中 非常 关键 的 一 个 功能 Ramm 


块 。 购 物 车 的 主要 功能 是 保留 用 户 选择 的 商品 信息 ， 下 一 生生 
用 户 可 以 在 购物 车 内 设置 选 购 商 品 的 数量 、 显 示 选 购 ”5 “mE 
商品 的 总 金额 ， 还 可 以 清除 选择 的 全 部 商品 信息 ， 重 "wx 上 a 

全 选 反 丢 pS 弄 [5] [al 共计 ， 16342,2 元 


新 选择 商品 信息 。 购 物 车 页 面 的 运行 结果 如 图 26.20 
所 示 。 图 26.20 购物 车 页 面 


购物 车 模块 主要 实现 添加 商品 、 删 除 商品 和 更 改 数 量 等 操作 。 
26.9.2 ”购物 车 中 商品 添加 技术 


购物 车 功能 的 实现 最 关键 的 部 分 就 是 如 何 将 商品 添加 到 购物 车 ， 如 果 不 能 完成 商品 的 添加 ， 那 么 购物 
车 中 的 其 他 操作 都 没有 任何 意义 。 

(1) 在 商品 显示 模块 中 ， 单 击 商品 中 的 “购买 ”按钮 ， 将 商品 放 到 购物 车 中 ， 并 进入 到 购物 车 页 面 。 
单 击 “ 购 买 ”按钮 调用 buycommo0 函 数 ， 购 买 商品 的 id 是 该 函数 的 唯一 参数 ， 在 buycommo0) 函 数 中 通过 
xmlhttp 对 象 调用 chklogin.php 文件 ， 并 根据 回 传 值 做 出 相应 处 理 。buycommo0 函 数 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\26\js\showcommo.js) 


六 
* 添 加 商品 ， 同 时 检查 用 户 是 否 登录 、 商 品 是 否 重复 等 
el 


function buycommo(keyX{ 


@ 
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”根据 商品 ID， 生 成 url Ww 
var url = "chklogin.php?key="+key; 
”使 用 xmlhttp 对 象 调用 chklogin.php 页 */ 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = function(){ 
if(xmlhttp.readyState == 4){ 
var msg = xmlhttp.responseText; 
广 。 用 户 没有 登录 */ 
imsg == 2 
alert(' 请 您 先 登录 '); 
return false; 
jelse if(msg == '3){ 
六 商品 已 添加 */ 
alert(' 该 商品 已 添加 '); 
return false; 
}else{ 
六 ”显示 购物 车 */ 
location='index.php?page=shopcar ; 
} 


b 
} 
xmlhttp.send(null); 


(2) 在 chklogin.php 文件 中 将 商品 添加 到 购物 车 中 。chklogin.php 页 的 代码 如 下 : 
《代码 位 置 ， 光盘 \TMNInstance\26\chklogin.php) 


<?php 
session_start(); 
header ( "Content-type: text/html; charset=UTF-8" ); // 设 置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 
Ar 

* 1 表示 添加 成 功 

* 2 表示 用 户 没有 登录 

* 3 表示 商品 已 添加 过 

* 4 表示 添加 时 出 现 错误 

* 5 表示 没有 商品 添加 
而 
S$reback = '0'; 
if(lempty($_SESSION['member'])X{ 

Sreback = 2"; 
Jelse{ 

Skey = $_GET[key"]; 

if($key == "}{ 

Sreback = '5"; 
jelse{ 
$boo = false; 


$sqls = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
$shopcont = $admindb->ExecSQL($sqls,$conn); 
if(lempty($shopcont[O][shopping'"])}X{ 
S$arr = explode('@', $shopcont[O][shopping"]); 
foreach($arr as $valuej{ 
$arrtmp = explode(,,$value); 
if($key == $arrtmp[O]X 
S$reback = '3"; 
$boo = true; 
break; 
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} 


} 
if($boo == false}{ 
$shopcont[0l['shopping] .= '@'. $key.",1’; 
$update = "update tb_user set shopping=".$shopcont[0]['shopping].”where name = 
™$_SESSION[member].™; 
$shop = $admindb->ExecSQL($update,$conn); 


if($shop){ 
Sreback = 1; 
}else{ 
Sreback = "4"; 
》 
} 
}else{ 


Stmparr = $key.",1"; 
S$updates = "update tb_user set shopping=".$tmparr." where name = ".$_SESSION[member].""; 
Sresult = $admindb->ExecSQL($updates,$conn); 


if($result)}{ 
Sreback = 1; 
Jelse{ 
S$reback = '4' 
0 
} 
} 
上 
echo S$reback; 
?> 


通过 分 析 上 述 代码 可 知 ，shopping 字段 保存 的 是 购物 车 中 的 商品 信息 ， 一 条 商品 信息 包括 两 部 分 ， 即 商 
品 id 和 商品 数量 ， 其 中 商品 数量 默认 为 1。 两 部 分 之 间 使 用 逗号 〈,) 分 隔 ， 如 果 添 加 多 个 商品 ， 则 每 个 商 
品 之 间 使 用 “@” 分 隔 。 

成 功 完成 商品 的 添加 操作 后 ， 即 可 进入 到 购物 车 页 面 ， 执 行 其 他 操作 。 


26.9.3 ”购物 车 展示 


购物 车 页 面 分 PHP 代码 页 和 Smarty 模板 页 。 在 PHP 代码 页 中 ， 首 先 读 取 tb_user 数据 表 中 shopping 字 
段 的 内 容 ， 如 果 字 段 为 室 ， 则 输出 “和 暂 无 商品 ”;， 如 果 数 据 库 中 有 数据 ， 则 循环 输出 数据 ， 并 将 商品 信息 
保存 到 数组 中 ， 再 传 给 模板 页 。 购 物 车 页 面 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TMNInstance\26myshopcar.php) 

<?php 

$select = "select id,shopping from tb_user where name =".$_SESSION[member]."…"'; 

Srst = $admindb->ExecSQL($select, $conn); 

if($rst[O['shopping"]=="™"X{ 

echo "<script>alert(' 购 物 车 中 暂时 没有 商品 !");window.location.href='index.php';</script>"; 


$commarr = array(); 
foreach($rst[0] as $valueX{ 
S$tmpnum = explode('@', $value); 
$shopnum = count($tmpnum); // 商 品类 数 
$sum = 0; 
foreach($tmpnum as $key => $vIX 
$s_commo = explode(',", $vl); 
$sql2 = "select id,name,m_price,fold,v_price from tb_commo"; 
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$commsql = $sql2." where id = ".$s_commo[0]; 
$arr = $admindb->ExecSQL($commsql,$conn); 
@$arr[O][num'] = $s_commo[1]; 
@$arr[ol[total] = $s_commo[1]*$arr[O][v_price']; 
$sum += $arr[0][total]: 
$commarr[$key] = $arr[0]; 
} 
下 
$smarty->assign('shoparr',$shopnum); 
$smarty->assign(commarr,$commarr); 
$smarty->assign('sum',$sum); 
?> 
商品 的 模板 页 不 仅 要 负责 用 户 购买 商品 信息 的 输出 ， 而 且 还 要 提供 可 以 对 商品 进行 修改 、 删 除 等 操作 
的 事件 接口 。 模 板 页 的 代码 如 下 : 
《代码 位 置 ， 光盘 \TMNVInstance\26\myshopcar.php) 
<table border="0" cellspacing="0" cellpadding="0" align="center"> 
<form id="myshopcar" name="myshopcar" method="post" action="#"> 
<tr> 
<td height="30" colspan="7" align="center" valign="middle" class="first"> 我 的 购物 车 </td> 
</tr> 
<tr> 
<td width="35" height="25" align="center" valign="middle" class="left">&nbsp;</td> 
<td width="100" height="25" align="center" valign="middle" class="center"> 商 品名 称 </td> 
<td width="100" height="25" align="center" valign="middle" class="center"> 购 买 数 量 </td> 
<td width="100" height="25" align="center" valign="middle" class="center"> 市 场 价 格 </td> 
<td width="100" height="25" align="center" valign="middle" class="center"> 会 员 价 格 </td> 
<td width="100" height="25" align="center" valign="middle" class="center"> 折 扣 率 </td> 
<td width="100" height="25" align="center" valign="middle" class="right"> 合 计 </td> 
</tr> 
{foreach key=key item=item from=$commarr} 
<tr> 
<td height="25" align="center" valign="middle" class="left"><input id="chk" name="chk[" type="checkbox" 
value="{$item.id}"></td> 
<td height="25" align="center” valign="middle” class="center"><div id = "c_name{$key}"> &nbsp; 
{$item.namej</div></td> 
<td height="25" align="center valign="middle" class="center"><input id="cnum{$key}" name="cnum{$key}" 
type="text" class="shorttxt" value="{$item.num}" onkeyup="cvp({$key},{$item.v_price},{$shoparr})"></td> 
<td height="25" align="center valign="middle" class="center"><div id="m_price{$key}">&nbsp;{$item.m_price} 
</div></td> 
<td height="25" align="center valign="middle" class="center"><div id="V_price{$key}">&nbsp;,{$item.v_price} 
</div></td> 
<td height="25" align="center” valign="middle"” class="center"><div id="fold{$key}">&nbsp;{Sitem.fold} 
</div></td> 
<td height="25" align="center” valign="middle"” class="right"><div id="total{$key}">&nbsp;{$item.total} 
</div></td> 
</tr> 
{/foreach} 
<tr> 
<td height="25" colspan="3" align="left" valign="middle"> 
<a href="#" onclick="return alldel(myshopcar)"> 全 选 </a> <a href="#" onclick="return overdel(myshopcar);"> 
反选 </a>&nbsp;&nbsp; 
<input type="button”value=" 删 除 选择 ”class="btn”" style="border-color: #FFFFFF;" onClick = 'return 


del(myshopcar);'> 
@ 


&nbsp;&nbsp; </td> 
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<td height="25" align="center”valign="middle"><input id="cont" name="cont" type="button”class="btn" 
value=" 继 续 购物 " onclick="returm conshop(myshopcar)" /></td> 

<td height="25" align="center" valign="middle"><input id="uid" name="uid" type="hidden" 
value="{$smarty.session.member}" ><input id="settle" name="settle" type="button" class="btn" value=" 去 收银 台 
" onclick="return formset(form)" /></td> 

<td height="25" colspan="2" align="right" valign="middle"><div id='sum'> 共 计 : {$sum}&nbsp; 元 </div></td> 

</tr> 
</form> 
</table> 


26.9.4 ”更 改 商 品 数 量 


对 于 新 添加 的 商品 ， 默 认 的 购买 数量 为 1， 在 购物 车 页 面 可 以 对 商品 的 数量 进行 修改 。 当 商品 数量 发 生 
变化 时 商品 的 “合计 ”金额 和 商品 总 金额 会 自动 发 生 改变 ,该 功能 是 通过 触发 text 文本 域 的 onkeyup 事件 调 
用 cvp0 函 数 实现 的 。cvpO 函 数 有 3 个 参数 ， 分 别 是 商品 id、 商 品 单价 和 商品 类 别 。 

首先 ， 通 过 商品 的 id 可 以 得 到 要 修改 商品 的 相关 表单 和 标签 属性 。 然 后 ， 通 过 商品 单价 和 输入 的 商品 
数量 计算 该 商品 的 合计 金额 。 接着 , 使 用 for 循环 得 到 其 他 商品 的 合计 金额 。 最 后 , 将 所 有 的 合计 金额 累加 ， 
并 输出 到 购物 车 页 面 。cvp0 函 数 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
function cvp(key,vpr,shoparrj{ 
var n_pre = 'total'; 
var num = 'cnum'+key.toString(); 
var total = n_pre+key.toString(); 
vart_number = document.getElementByld(num).value; 
var ttl = t_number * vpr; 
document.getElementByld(total).innerHTML = tl; 
var sm = 0; 


for(var i = 0; i < shoparr; i++}{ 


var aaa = document.getElementByld(n_pre+i.toString()).innerText; 
sm += parselnt(aaa); 


上 
document.getElementByld(sum').innerHTML = ' 共 计 : '+sm+' 元 '; 


1 
这 里 所 更 改 的 商品 数量 ， 并 没有 被 保存 到 数据 库 中 ， 如 果 希 望 保存 ， 那 么 单 击 “ 继 续 购 物 ” 按 钮 ， 则 
可 以 将 商品 数量 更 新 到 数据 库 中 。 


26.9.5 ”删除 商品 


当 对 添加 的 商品 不 满意 时 ， 可 以 对 商品 进行 删除 操作 。 操 作 流程 为 : 首先 选中 要 删除 的 商品 前 面 的 复 
选 框 ， 如 果 全 部 删除 ， 则 可 以 单 击 “ 全 选 ” 按 钮 或 “反选 ”按钮 。 然 后 再 单 击 “ 删 除 选择 ”按钮 ， 在 弹出 
的 警告 框 中 单 击 “ 确 定 ”按钮 ， 商 品 将 被 全 部 删除 。 删 除 商品 的 页 面 的 运行 结果 如 图 26.21 所 示 。 

所 有 的 删除 操作 都 是 通过 JS 脚本 文件 shopcarjs 来 实现 的 ， 相 关 的 函数 包括 alldel0、overdel0 和 del0。 

alldel0 和 overdel0 函 数 实现 的 原理 比较 简单 ， 通 过 触发 onclick 事件 来 改变 复 选 框 的 选中 状态 。 函 数 代 
码 如 下 : 


@_ 
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商品 大 和 购 于 东 重 
上 日。 7 
卓 WEE 1 
四 数码 相机 1 
国信 车 
全 达 巨 闪 那 际 鞍 拓 元 
高 品名 称 隐 忒 热 重 市 场 价格 
加 守 屋 时 院 hn 483 
流 最 显示 器 上 S83 
四 类 码 相机 1 ee 
全 过 区 并 央视 & 择 续 坟 风 芒 元 


26.21 删除 商品 流程 


(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 

// 全 部 选择 /取消 
function alldel(formX{ 

var leng = form.chk.length; 

ifleng==undefined){ 

iform.chk.checked) 
form.chk.checked=true; 
}else{ 
for( vari= 0; i < leng; i++) 


if(!form.chk[il.checked) 
form.chk[i].checked = true; 
} 


return false; 


} 
/反选 
function overdel(form){ 
varleng = form.chk.length; 
if(lleng==undefined){ 
if(!form.chk.checked) 
form.chk.checked=true; 
else 
form.chk.checked=false; 
Jelse{ 
for( vari= 0; i < leng; i++) 


if(!form.chk[li].checked) 
form.chk[i].checked = true; 
else 
form.chk[i].checked = false; 


return false; 


} 
使 用 alldel0 或 overdel0 选 中 复 选 框 后 , 即 可 调用 del0 函 数 来 实现 删除 功能 。 del0 函 数 首先 使 用 for 循环 ， 
将 被 选中 的 复 选 框 的 value 值 取出 并 存 成 数组 ， 然 后 根据 数组 生成 url， 并 使 用 xmlhttp 对 象 调用 这 个 url， 


当 处 理 完 毕 后 ， 根 据 返回 值 弹出 提示 或 刷新 本 页 。 该 函数 的 代码 如 下 : 
〈 代 码 位 置 : 光盘 \TMNVInstance\26\js\vshopcar.js) 


全 
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/删除 记录 */ 
function del(form){ 
if(Iwindow.confirm(' 是 否 要 删除 数据 ??')X{ 


Jelse{ 
var leng = form.chk.length; 
if(leng==undefinedX{ 
if(!form.chk.checkedX{ 
alert(' 请 选取 要 删除 数据 !"); 
Jelse{ 


rd = form.chk.value; 
var url = 'delshop.php?rd="+rd; 
xmlhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = delnow; 
xmlhttp.send(null); 

’ 

jelsef 

var rd=new Array(); 

varj=0; 

for( vari = 0; i < leng, i++) 


if(form.chk[i].checkedX{ 
rd[j++] = form.chkli].value; 
1 


} 

if(rd == "){ 
alert(' 请 选取 要 删除 数据 !"); 

}else{ 
var url = "delshop.php?rd="+rd; 
xmilhttp.open("GET",url,true); 
xmlhttp.onreadystatechange = delnow; 
xmlhttp.send(null); 


} 
} 
return false; 


} 
function delnow(X{ 
if(xmlhttp.readyState == 4){ 
if(xmlhttp.status == 200){ 
var msg = xmlhttp.responseText; 
if(msg != "1'X{ 
alert(' 删 除 失败 '+msg); 
}else{ 
alert( 删除 成 功 ); 
location=('?page=shopcar); 


} 


26.9.6 ”保存 购物 车 


当 用 户 希 望 保存 商品 更 改 后 的 商品 数量 时 ,可 以 单 击 “ 继 续 购物 ”按钮 ,将 触发 onclick 事件 调用 conshopO 
函数 保存 数据 ， 该 函数 有 一 个 参数 ， 就 是 当前 表单 的 名 称 。 在 conshop0 函 数 内 ， 根 据 复 选 枉 和 商品 数量 文 


@ 
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本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 
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本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 
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本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 
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本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 
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本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 
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本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
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xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 


第 26 章 ”综合 实例 (五 ) 一 一 电子 商务 网 站 


本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 


第 26 章 ”综合 实例 (五 ) 一 一 电子 商务 网 站 


本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 


第 26 章 ”综合 实例 (五 ) 一 一 电子 商务 网 站 


本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 


第 26 章 ”综合 实例 (五 ) 一 一 电子 商务 网 站 


本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 


第 26 章 ”综合 实例 (五 ) 一 一 电子 商务 网 站 


本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 


第 26 章 ”综合 实例 (五 ) 一 一 电子 商务 网 站 


本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 


第 26 章 ”综合 实例 (五 ) 一 一 电子 商务 网 站 


本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 
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本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 
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本 域 ， 生 成 两 个 数组 ft 和 snd， 分 别 保存 商品 id 和 商品 数量 。 
这 里 要 注意 ， 两 个 数组 的 值 是 要 相互 对 应 的 ， 如 商品 1 的 id 保存 到 fst[1] 中 ， 那 么 商品 1 的 数量 就 要 保 
存 到 snd[1] 中 ， 然 后 根据 这 两 个 数组 生成 一 个 url， 使 用 xmlhttprequest 对 象 调用 url， 最 后 根据 回 传 信息 作 
出 相应 的 判断 。conshopO 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
// 更 改 商 品 数量 
function conshop(formj{ 
varn_pre = 'cnum'; 
var lang = form.chk.length; 
if(lang == undefined}{ 
var fst = form.chk.value; 
var snd = form.cnum0.value; 
Jelse{ 
var fst= new Array(); 
var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = nN_pre+i.toString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 


return false; 

1 

snd[i] = stmp; 

var ftmp = form.chk[i].value; 

fst[i] = ftmp; 

} 

} 
var url = 'changecar.php?fst='+fst+'&snd='+snd; 
xmlhttp.open("GET",url,true); 


xmlhttp.onreadystatechange = updatecar; 
xmlhttp.send(null); 


} 
function updatecar(){ 
if(xmlhttp.readyState == 4X{ 
var msg = xmlhttp.responseText; 
if(msg == "1'X{ 
location='index.php'; 
jelse{ 
alert( 操作 失败 +msg); 


} 


术 
在 conshop0) 函 数 中 调用 的 changecar.php 页 为 数据 处 理 页 ， 该 页 将 商 
保存 到 shopping 字段 内 。 该 页 面 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance26\changecar.php) 


id 和 商品 数量 进行 重新 排列 ， 并 


缉 


<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); ll 设置 文件 编码 格式 
require("system/system.inc.php"); /包含 配置 文件 


$sql = "select id,shopping from tb_user where name = ".$_SESSION[member].""; 
Srst = $admindb->ExecSQL($sql,$conn); 
Sreback = '0'; 


PHP+MySQL 开发 实战 


$changecar = array(); 
if(isset($_GETT'fst]) && isset($_GET[snd]))f 
Sfst = $_GETIYfst]; 
$snd =$_GET[snd]; 
S$farr = explode(',', $fst); 
S$sarr = explode(",', $snd); 
S$upcar = array(); 
for($i = 0; $i < count($farr); $i++){ 
$upcarf$i] = $farr[$i].",".$sarr[$i]; 
y 
if(count($farr) > 1X{ 
Supdate = "update tb_user set shopping=".implode('@',$upcar)." where name = 
".$_SESSION[member].”"'; 
Jelse{ 
Supdate = "update tb_user set shopping=".$upcar[0]." where name = ".$_SESSION[member].""'; 


} 
$shop = $admindb->ExecSQL($update, $conn); 


26.10 ”收银 台 模 块 设计 
茹 4 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 收 银 台 模块 设计 .exe 


26.10.1 收银 台 模块 概述 


当 用 户 停止 浏览 商品 准备 结账 时 , 可 以 单 击 购物 车 页 面 
中 的 “去 收银 台 ”按钮 , 该 按钮 将 触发 onclick 事件 调用 formset0 
函数 显示 订单 页 面 ， 当 用 户 提交 订单 后 ， 系 统 将 订单 保存 到 数 
据 表 tb_form 中 ， 同 时 清空 购物 车 ， 并 显示 订单 信息 提醒 用 户 
记录 订单 号 。 当 货款 发 出 后 ， 还 可 以 对 订单 进行 查询 。 收 银 台 
页 面 的 运行 结果 如 图 26.22 所 示 。 

本 节 所 涉及 的 页 面 有 显示 订单 (formset0 函 数 ) 、 填 写 订 
单 (settle.php、settle.tpl) 、 提 交 订 单 (settle_ chk.php) 、 反 
馈 订单 forminfo.php、forminfo.tpl) 和 查询 订单 5 部 分 。 


26.10.2 ”PDO 操作 MySQL 数据 库 技术 


在 收银 台 模 块 中 ， 通 过 PDO 中 的 方法 完成 订单 信息 的 汪 ON 
加 和 数据 的 更 新 操作 。 同 样 调用 数据 库 管理 关 AdminDB 中 的 ExeesQL 0 方法， 完成 数据 的 添加 操作 。 


@ 


PHP+MySQL 开发 实战 


$changecar = array(); 
if(isset($_GETT'fst]) && isset($_GET[snd]))f 
Sfst = $_GETIYfst]; 
$snd =$_GET[snd]; 
S$farr = explode(',', $fst); 
S$sarr = explode(",', $snd); 
S$upcar = array(); 
for($i = 0; $i < count($farr); $i++){ 
$upcarf$i] = $farr[$i].",".$sarr[$i]; 
y 
if(count($farr) > 1X{ 
Supdate = "update tb_user set shopping=".implode('@',$upcar)." where name = 
".$_SESSION[member].”"'; 
Jelse{ 
Supdate = "update tb_user set shopping=".$upcar[0]." where name = ".$_SESSION[member].""'; 


} 
$shop = $admindb->ExecSQL($update, $conn); 


26.10 ”收银 台 模 块 设计 
茹 4 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 收 银 台 模块 设计 .exe 


26.10.1 收银 台 模块 概述 


当 用 户 停止 浏览 商品 准备 结账 时 , 可 以 单 击 购物 车 页 面 
中 的 “去 收银 台 ”按钮 , 该 按钮 将 触发 onclick 事件 调用 formset0 
函数 显示 订单 页 面 ， 当 用 户 提交 订单 后 ， 系 统 将 订单 保存 到 数 
据 表 tb_form 中 ， 同 时 清空 购物 车 ， 并 显示 订单 信息 提醒 用 户 
记录 订单 号 。 当 货款 发 出 后 ， 还 可 以 对 订单 进行 查询 。 收 银 台 
页 面 的 运行 结果 如 图 26.22 所 示 。 

本 节 所 涉及 的 页 面 有 显示 订单 (formset0 函 数 ) 、 填 写 订 
单 (settle.php、settle.tpl) 、 提 交 订 单 (settle_ chk.php) 、 反 
馈 订单 forminfo.php、forminfo.tpl) 和 查询 订单 5 部 分 。 


26.10.2 ”PDO 操作 MySQL 数据 库 技术 


在 收银 台 模 块 中 ， 通 过 PDO 中 的 方法 完成 订单 信息 的 汪 ON 
加 和 数据 的 更 新 操作 。 同 样 调用 数据 库 管理 关 AdminDB 中 的 ExeesQL 0 方法， 完成 数据 的 添加 操作 。 


@ 
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26.10.3 显示 订单 


订单 信息 提交 页 面 的 输出 由 formsetO 函 数 决定 ， 它 将 商品 信息 整理 ， 通 过 open 方法 打开 settle.php 页 来 
显示 订单 ， 并 将 整理 后 的 商品 信息 传递 到 settle .php 文件 中 。formset0 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
function formset(form}{ 
var uid = form.uid.value; 
varn_pre = 'cnum'; /数量 
var lang = form.chk.length; 
if(lang == undefined){ 


var fst = form.chk.value; /商品 id 

var snd = form.cnum0.value; /购买 数量 
Jelse{ 

var fst= new Array(); 


var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = n_pre+itoString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 
return false; 


snd[i] = stmp; 
var ftmp = form.chk[i].value; 
fst[i] = ftmp; 

} 


} 
open('settle.php?uid='+uid+'&fst="+fst+'&snd='+snd,'_blank','width=500 height=450',false); 


p94 
a 说 明 因为 open 方法 使 用 了 _blank 参数 来 打开 一 个 新 的 页 面 ，session 值 传 不 过 去 ， 所 以 这 里 使 用 隐 
藏 域 来 传递 用 户 名 称 。 


26.10.4 填写 订单 


settle.php 直接 将 接收 的 值 传 给 settle.tpl 模板 ， 并 载 入 settle.tpl 模板 。settle.php 页 面 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\InstanceW26\settle.php) 

<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); /1 设置 文件 编码 格式 

require("system/system.inc.php"); // 包 含 配置 文件 

S$fst = $_GETTfst]; 

$snd=$_GET[snd]; 

S$uid =$_GET[uid]; 

$smarty->assign('title',' 收 银 台 "); 

$smarty->assign('fst', $fst); 

$smarty->assign('snd',$snd); 

$smarty->assign('uid', $uid); 
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26.10.3 显示 订单 


订单 信息 提交 页 面 的 输出 由 formsetO 函 数 决定 ， 它 将 商品 信息 整理 ， 通 过 open 方法 打开 settle.php 页 来 
显示 订单 ， 并 将 整理 后 的 商品 信息 传递 到 settle .php 文件 中 。formset0 函 数 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\26\js\shopcar.js) 
function formset(form}{ 
var uid = form.uid.value; 
varn_pre = 'cnum'; /数量 
var lang = form.chk.length; 
if(lang == undefined){ 


var fst = form.chk.value; /商品 id 

var snd = form.cnum0.value; /购买 数量 
Jelse{ 

var fst= new Array(); 


var snd = new Array(); 
for(vari = 0; i < lang; i++X{ 
var nm = n_pre+itoString(); 
var stmp = document.getElementByld(nm).value; 
if(stmp =="||isNaN(stmp)X{ 
alert(' 不 允许 为 空 、 必 须 为 数字 '); 
document.getElementByld(nm).select(); 
return false; 


snd[i] = stmp; 
var ftmp = form.chk[i].value; 
fst[i] = ftmp; 

} 


} 
open('settle.php?uid='+uid+'&fst="+fst+'&snd='+snd,'_blank','width=500 height=450',false); 


p94 
a 说 明 因为 open 方法 使 用 了 _blank 参数 来 打开 一 个 新 的 页 面 ，session 值 传 不 过 去 ， 所 以 这 里 使 用 隐 
藏 域 来 传递 用 户 名 称 。 


26.10.4 填写 订单 


settle.php 直接 将 接收 的 值 传 给 settle.tpl 模板 ， 并 载 入 settle.tpl 模板 。settle.php 页 面 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\InstanceW26\settle.php) 

<?php 

session_start(); 

header ( "Content-type: text/html; charset=UTF-8" ); /1 设置 文件 编码 格式 

require("system/system.inc.php"); // 包 含 配置 文件 

S$fst = $_GETTfst]; 

$snd=$_GET[snd]; 

S$uid =$_GET[uid]; 

$smarty->assign('title',' 收 银 台 "); 

$smarty->assign('fst', $fst); 

$smarty->assign('snd',$snd); 

$smarty->assign('uid', $uid); 


PHP+MySQL 开发 实战 


$smarty->display('settle tpl); 
?> 


settle tpl 模板 显示 一 个 表单 , 这 个 表单 的 内 容 需 要 用 户 来 填写 , 包括 收 货 人 、 联 系 电话 等 信息 。 而 从 PHP 


页 传 过 来 的 几 个 变量 则 被 保存 到 隐藏 域 以 传递 到 处 理 页 ， 在 表单 中 将 数据 提交 到 settle_ chk.php 处 理 页 。 


26 


.10.5 “处 理 订单 


处 理 页 settle_chk.php 获取 表单 中 提交 的 数据 ， 根 据 用 户 提交 的 商品 信息 ， 重 新 查找 数据 表 tb_commo， 


并 从 数据 表 中 提取 商品 信息 ， 保 存 到 数组 中 ， 然 后 处 理 页 将 数组 作为 一 条 记录 添加 到 表 tb_form 内 。 


数据 添加 成 功 的 同时 ， 处 理 页 会 根据 uid 找到 该 用 户 ， 将 shopping 字段 清空 ， 最 后 调用 forminfo.php 页 


来 显示 新 添加 的 订单 信息 。settle_chk.php 页 的 代码 如 下 : 


610 


(代码 位 置 ， 光盘 \TM\Instance\26\settle_chk.php) 


<?php 
header ( "Content-type: text/html; charset=UTF-8" ); /设置 文件 编码 格式 
require("system/system.inc.php"); // 包 含 配置 文件 


$sql="insert into tb_form(formid,commo_id,commo_name,commo_num,agoprice,fold,total,vendee,taker,address, 
tel,code, pay_method,del_method,formtime,state)values("; 
S$formid=time(); 
S$tmpid = explode(',',$_POST['fst"]); 
$tmpnm = explode(,,$_POST[snd]); 
S$number = count($tmpid); 
S$tmpna = array(); 
S$tmpvp = array(); 
S$tmpfd = array(); 
S$tmptt = 0; 
if($number >1){ 
for($i = 0; $i < $number; $i++}{ 
$tmpsql = "select name,v_price,fold from tb_commo where id = ".$tmpid[$i].""; 
S$tmprst = $admindb->ExecSQL($tmpsql,$conn); 
S$tmpna[$i] = $tmprst[O]['name']; 
S$tmpvp[$i] = $tmprst[0][v_price]; 
S$tmpfd[$i] = $tmprst[O[fold]; 
$tmptt += $tmprst[O][v_price]* $tmpnm[$i]; 
@$tmpsell = $tmprst[O]['sell] + 1; 
$addsql = "update tb_commo set sell = ".$tmpsell.” where id = ".$tmpid[$i]."™; 
$addrst = $admindb->ExecSQL($addsql,$conn); 


} 

$sql.="".$formid.",".$_POSTT[fst].",".implode(',,$tmpna).",".$_POST[snd'].",".implode(",',$tmpvp).",".im 
plode(',,$tmpfd).",".$tmptt™,".$_POST[uid].™"; 
jelse if($number == 1){ 

$tmpsql = "select name,v_price,fold from tb_commo where id = ".$tmpid[0].""; 

S$tmprst = $admindb->ExecSQL($tmpsql,$conn); 

S$tmptt= $tmprst[0][v_price] * $tmpnm[O]; 

@$tmpsell = $tmprst[OJ['sell] + 1; 

$addsql = "update tb_commo set sell = ".$tmpsell." where id = ".$tmpid[0].™; 

S$addrst = $admindb->ExecSQL($addsql,$conn); 

$sql.="".$formid.",".$_POST[fst].",". ee ™™".$_ POST[snd].",".$tmprst[O]['v_price’].",".$t 
mprstlOjrfold]",".$tmptt",w$_POSTIuid] 
}else{ 

echo 'error ; 

exit(); 


} 
$sql.=",".$_POST[taker].",".$_POST[address].",”".$_POSTTtel].”",".$_POST[code].”",”.$_POST[pay].",”".$ 
_POST[del].",".date("Y-m-d Hii:s").",0)"; 
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S$InsertSQL = $admindb->ExecSQL($sql,$conn); 
if(false == $lnsertSQL){ 
echo "<script>alert(' 购 买 失 败 ');history.back;</script>"; 
jelset 
S$updsql = "update tb_user set consume=".$tmptt.",shopping=" where name = ".$_POST[uid].""; 
S$updrst = $admindb->ExecSQL($updsql,$conn); 
echo "<script>top.opener.location.reload();</script>"; 
echo "<script>open(forminfo.php?fid=$formid,，blank','width=750 height=650',false);window.close();</script>"; 
上 


?> 


由 于 篇 幅 所 限 ， 有 关 反 馈 订单 和 查询 订单 的 内 容 这 里 不 再 讲解 ， 请 读者 参考 本 书 光盘 中 的 源 代码 。 
26.11 后 台 首 页 设计 
区 4 视频 讲解 :光盘 \TM\Video\ 第 26 章 \ 后 台 首页 设计 .exe 


26.11.1 后 台 首页 概述 


后 台 管 理 系统 是 网 站 管理 员 对 商品 、 会 员 及 公告 等 信息 进行 统一 管理 的 场所 ， 本 系统 的 后 台 主要 包括 
以 下 功能 。 

类 别管 理 模块 : 主要 包括 对 商品 类 别 的 添加 、 修 改 及 删除 操作 。 

商品 管理 模块 : 主要 包括 对 商品 的 添加 、 修 改 、 删 除 及 订单 处 理 。 

fhe 主要 包括 管理 员 管 理 和 会 员 管 理 。 其 中 管理 员 管理 是 实现 对 管理 员 的 添加 、 删 除 
修改 功能 ， 会 员 管理 则 包括 删除 和 冻结 功能 。 

回 pe i 管理 模块 ， 主 要 包括 公告 的 添加 及 删除 操作 。 

链接 管理 模块 : 主要 包括 添加 、 修 改 和 删除 友情 链接 。 

后 台 首 页 的 运行 结果 如 图 26.23 所 示 。 


图 26.23 后 台 首页 运行 的 结果 


26.11.2 ”后 台 网 页 布局 技术 


后 台 首 页 和 前 台 首 页 不 同 ， 其 使 用 的 是 框架 布局 。 框 架 布局 的 特点 是 : 可 以 将 容器 窗口 划分 为 若干 个 
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S$InsertSQL = $admindb->ExecSQL($sql,$conn); 
if(false == $lnsertSQL){ 
echo "<script>alert(' 购 买 失 败 ');history.back;</script>"; 
jelset 
S$updsql = "update tb_user set consume=".$tmptt.",shopping=" where name = ".$_POST[uid].""; 
S$updrst = $admindb->ExecSQL($updsql,$conn); 
echo "<script>top.opener.location.reload();</script>"; 
echo "<script>open(forminfo.php?fid=$formid,，blank','width=750 height=650',false);window.close();</script>"; 
上 


?> 


由 于 篇 幅 所 限 ， 有 关 反 馈 订单 和 查询 订单 的 内 容 这 里 不 再 讲解 ， 请 读者 参考 本 书 光盘 中 的 源 代码 。 
26.11 后 台 首 页 设计 
区 4 视频 讲解 :光盘 \TM\Video\ 第 26 章 \ 后 台 首页 设计 .exe 


26.11.1 后 台 首页 概述 


后 台 管 理 系统 是 网 站 管理 员 对 商品 、 会 员 及 公告 等 信息 进行 统一 管理 的 场所 ， 本 系统 的 后 台 主要 包括 
以 下 功能 。 

类 别管 理 模块 : 主要 包括 对 商品 类 别 的 添加 、 修 改 及 删除 操作 。 

商品 管理 模块 : 主要 包括 对 商品 的 添加 、 修 改 、 删 除 及 订单 处 理 。 

fhe 主要 包括 管理 员 管 理 和 会 员 管 理 。 其 中 管理 员 管理 是 实现 对 管理 员 的 添加 、 删 除 
修改 功能 ， 会 员 管理 则 包括 删除 和 冻结 功能 。 

回 pe i 管理 模块 ， 主 要 包括 公告 的 添加 及 删除 操作 。 

链接 管理 模块 : 主要 包括 添加 、 修 改 和 删除 友情 链接 。 

后 台 首 页 的 运行 结果 如 图 26.23 所 示 。 


图 26.23 后 台 首页 运行 的 结果 


26.11.2 ”后 台 网 页 布局 技术 


后 台 首 页 和 前 台 首 页 不 同 ， 其 使 用 的 是 框架 布局 。 框 架 布局 的 特点 是 : 可 以 将 容器 窗口 划分 为 若干 个 
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子 窗口 ， 每 个 子 窗口 可 以 分 别 显示 不 同 的 网 页 ， 网 页 之 间 为 相互 独立 的 ， 没 有 直接 的 关联 ， 又 由 一 个 网 页 
将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 者 的 浏览 器 中 。 框 架 布 局 的 好 处 是 : 每 次 浏览 者 发 出 
对 页 面 的 请 求 时 ， 只 下 载 发 生变 化 的 框架 页 面 ， 其 他 子 页 面 保持 不 变 。 下 面 来 具体 看 一 下 框架 布局 的 使 用 


格式 及 属性 。 


1. 框架 布局 格式 
框架 布局 的 格式 很 简单 ， 只 要 几 行 代码 即 可 。 常 用 的 格式 如 下 : 


<html> 
<head> 


</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 
<body> 
</body> 


</noframes> 
</html> 


其 中 <frameset> 和 <frame> 标 签 是 框架 集 标记 , 而 <noframes> 标 签 是 为 了 防止 浏览 器 不 支持 框架 而 实行 的 
-种 补救 措施 。 如 果 浏 览 器 不 支持 框架 集 ， 就 会 执行 <noframes> 标 签 里 的 内 容 ， 让 用 户 能 够 正常 浏览 网 页 。 
2. 框架 集 属性 
框架 集 包含 各 个 框架 的 信息 ， 通 过 <frameset> 标 签 来 定义 。 框 架 是 按照 行 和 列 来 组 织 的 ， 可 以 使 用 
<frameset> 标 签 的 属性 对 框架 的 结构 进行 设置 。 下 面 给 出 框架 集 的 常用 属性 值 、 说 明和 应 用 举例 ， 如 表 26.1 


所 示 。 
表 26.1 框架 集 的 常用 属性 
参数 说 明 描述 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， | <frameset cols="25%.100.*" > 
COLS 取 值 有 3 种 形式 ， 像 素 、 百 分 比 (%) 和 | <frame></frame> 
相对 尺寸 〈*) </frameset> 
<frameset rows="25%,100,*" > 
Ns 在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， <frame> 
取 值 和 COLS 类 似 ， 也 是 3 种 形式 <frame> 
</frameset> 
ee 指定 框 加 周转 是否 显示 边框， 取信 为 1( 显 <framset cols="25%.*" cols="*" frameborder="0"> 
示 边 框 ， 默 认 值 ) 或 0 (不 显示 边框 
</frameset> 
ee 指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 默 <framset cols="25%.*" cols="*" framespacing="1"> 
认 是 无 间隔 的 
</frameset> 
<framset cols="25%.*" cols="*" frameborder="1" 
指定 边框 的 宽度 ，frameborder 属性 为 1 时 | border="5"> 
BORDER Es 


该 属性 才 有 效 


</frameset> 
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子 窗口 ， 每 个 子 窗口 可 以 分 别 显示 不 同 的 网 页 ， 网 页 之 间 为 相互 独立 的 ， 没 有 直接 的 关联 ， 又 由 一 个 网 页 
将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 者 的 浏览 器 中 。 框 架 布 局 的 好 处 是 : 每 次 浏览 者 发 出 
对 页 面 的 请 求 时 ， 只 下 载 发 生变 化 的 框架 页 面 ， 其 他 子 页 面 保持 不 变 。 下 面 来 具体 看 一 下 框架 布局 的 使 用 


格式 及 属性 。 


1. 框架 布局 格式 
框架 布局 的 格式 很 简单 ， 只 要 几 行 代码 即 可 。 常 用 的 格式 如 下 : 


<html> 
<head> 


</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 
<body> 
</body> 


</noframes> 
</html> 


其 中 <frameset> 和 <frame> 标 签 是 框架 集 标记 , 而 <noframes> 标 签 是 为 了 防止 浏览 器 不 支持 框架 而 实行 的 
-种 补救 措施 。 如 果 浏 览 器 不 支持 框架 集 ， 就 会 执行 <noframes> 标 签 里 的 内 容 ， 让 用 户 能 够 正常 浏览 网 页 。 
2. 框架 集 属性 
框架 集 包含 各 个 框架 的 信息 ， 通 过 <frameset> 标 签 来 定义 。 框 架 是 按照 行 和 列 来 组 织 的 ， 可 以 使 用 
<frameset> 标 签 的 属性 对 框架 的 结构 进行 设置 。 下 面 给 出 框架 集 的 常用 属性 值 、 说 明和 应 用 举例 ， 如 表 26.1 


所 示 。 
表 26.1 框架 集 的 常用 属性 
参数 说 明 描述 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， | <frameset cols="25%.100.*" > 
COLS 取 值 有 3 种 形式 ， 像 素 、 百 分 比 (%) 和 | <frame></frame> 
相对 尺寸 〈*) </frameset> 
<frameset rows="25%,100,*" > 
Ns 在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， <frame> 
取 值 和 COLS 类 似 ， 也 是 3 种 形式 <frame> 
</frameset> 
ee 指定 框 加 周转 是否 显示 边框， 取信 为 1( 显 <framset cols="25%.*" cols="*" frameborder="0"> 
示 边 框 ， 默 认 值 ) 或 0 (不 显示 边框 
</frameset> 
ee 指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 默 <framset cols="25%.*" cols="*" framespacing="1"> 
认 是 无 间隔 的 
</frameset> 
<framset cols="25%.*" cols="*" frameborder="1" 
指定 边框 的 宽度 ，frameborder 属性 为 1 时 | border="5"> 
BORDER Es 


该 属性 才 有 效 


</frameset> 
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3. 框架 属性 
使 用 <frame> 标 签 可 以 设置 框架 的 属性 ， 包 括 框架 的 名 称 ， 框 架 是 否 包含 滚动 条 以 及 在 框架 中 显示 的 网 


页 等 。<frame> 标 签 的 常用 属性 及 其 说 明 如 表 26.2 所 示 。 
表 26.2 框架 属性 
参数 说 有明 
NAME 指定 框架 的 名 称 
SRC 指定 在 框架 中 显示 的 网 页 文件 (包括 HMRL、PHP、JSP 等 网 页 文件 ) 
FRAMEBODER 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框 ， 为 默认 ) 或 0 (不 显示 边框 
NORESIZE 可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 
SCROLLING 指定 框架 是 否 包含 滚动 条 。 属 性 可 以 是 yes (有 ) 、no (没有 ) 和 auto (自由 ) 


26. 


11.3 ”后 台 首 页 实现 过 程 


(1) 定义 框架 页 面 main.tpl， 包 含 3 个 文件 : top.tpl、leftphp 和 defaultphp。main.tpl 页 的 代码 如 下 : 
《代码 位 置 : 光盘 \TM\Instance26\admin\system\templates\main.tpl ) 
<!IDOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/ 
xhtml1-transitional.dtd"> 
<html xmlns="http://www.w3.org/1999/xhtml"> 
<head> 
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> 
<title> 明 日 购物 商城 后 台 管 理 系统 </title> 
<link rel="stytlesheet" href="css/style.css" /> 
</head> 
<frameset rows="113,*,100" cols="1004" frameborder="no" border="0" framespacing="0"> 
<frame src="top.php" name="topFrame" scrolling="No" noresize="noresize" id="topFrame" title="topFrame" /> 
<frameset rows="*" cols="10%,210,*,10%" framespacing="0" frameborder="no" border="0"> 


<frame src="s.php" name="|Frame" frameborder="0" scrolling="auto" noresize="noresize” id="|Frame" 
title="leftFrame" /> 

<frame src="left.php" name="leftFrame" frameborder="0" scrolling="auto" noresize="noresize" 
id="leftFrame'" title="leftFrame" /> 

<frame src="default.php" name="mainFrame" id="mainFrame" title="mainFrame" /> 
<frame src="s.php" name="rFrame" frameborder="0" scrolling="auto" noresize="noresize” id="rFrame" 
tile="leftFrame" /> 

</frameset> 
<frame src="bottom.php” name="bottomFrame" scrolling="No" noresize="noresize" id="bottomFrame" 

title="bottomFrame" /> 
</frameset> 
<noframes><body> 
</body> 
</noframes> 
</html> 


(2) leftphp 页 是 一 个 树 形 菜单 ， 应 用 DIV+JavaScript+CSS 实现 。 首 先 介绍 div 标签 ， 在 lefttpl 模板 


文件 中 。 其 关键 代码 如 下 : 


(代码 位 置 ， 光 盘 \TM\Instance26\admin\system\templates\left.tp1) 
<!-- 载 入 CSS 样式 和 JavaScript 肢 本 --> 
<link href="css/left.css" rel="stylesheet" type="text/css" /> 
<script language="javascript" src="js/left.js"></script> 


<!- ”类别 管理 菜单 ， 注 意 加 粗 的 地 方 ”一 > 


全 
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<div id="type" align="center" onclick="javascriptchange(one,type):"> 类 别管 理 </div> 

<!-- 子 菜单 ”--> 

<div id="one" style="display: "> 

<div id="addtype" align="center"><a href="addtype.php" target="mainFrame" id="menu"> 添 加 类 别 </a></div> 
<div id="showtype" align="center"><a href="showtype.php" target="mainFrame" id="menu"> 查 看 类 别 </a></div> 
</div> 

<div id="hidediv" align="center"></div> 

<!- 商品 管理 菜单 ”一 > 

<div id="commo" align="center" onclick="javascript:change(two,type);"> 类 别管 理 </div> 

<div id="two"style="display:none"> 

<!-- 商品 管理 子 菜单 -> 


</div> 
SS 人 
反 除了 加 粗 的 这 名 称 和 js 事件 不 同 外 ， 其 他 菜单 的 结构 完全 相同 ， 此 时 只 需 修改 超 链接 即 可 。 


全 s 广 刘 除了 第 一 个 类 别 菜单 的 子 菜单 display 样式 为 空 外 ， 其 他 几 个 子 菜单 的 div 样式 都 为 “display= 


none;”。 


该 页 面 在 Dreamweaver 中 的 效果 如 图 26.24 所 示 。 
因为 其 他 子 菜单 的 样式 为 display=none， 所 以 只 有 “类 别管 


和 
区 吉 法 别 ， 


理 ” 子 菜单 是 可 见 的 ， 下 面 为 它 添加 js 事件 。leftjs 脚本 文件 的 代 
码 如 下 : ee 
(代码 位 置 ， 光盘 \TM\Instance\26\admin\js\left.js) 26.24 div 树 形 菜单 


function change(nu,IxX{ 
if(nu.style.display == "none"){ 
nu.style.display = "™"; 
Ix.style.background="url(images/admin(5).gif)"; 
}else{ 
nu.style.display = "none"; 
Ix.style.background="url(images/admin(1).gif)"; 
} 


时 

最 后 在 left.css 中 设置 div 的 长 、 宽 等 一 些 默 认 参 数 。 一 个 简单 而 又 实用 的 树 形 菜单 就 完成 了 。 对 于 后 
台 的 大 部 分 模块 来 说 ， 其 功能 实现 的 方法 和 开发 步 又 在 前 台 的 模块 设计 中 基本 都 已 经 介绍 过 。 由 于 篇 幅 所 
限 ， 这 里 不 再 对 后 台 管 理 模块 进行 详细 讲解 。 


20:12 小 结 


本 项 目 使 用 Smarty、PDO 数据 库 抽象 层 和 Ajax 等 目前 的 主流 技术 ， 开 发 一 个 电子 商务 平台 。 通 过 本 
章 的 学 习 ， 读 者 不 仅 可 以 了 解 电子 商务 网 站 的 开发 流程 ， 而 且 还 可 以 熟悉 购物 车 、 订 单 及 加 密 技 术 的 开发 
思想 。 
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<div id="type" align="center" onclick="javascriptchange(one,type):"> 类 别管 理 </div> 

<!-- 子 菜单 ”--> 

<div id="one" style="display: "> 

<div id="addtype" align="center"><a href="addtype.php" target="mainFrame" id="menu"> 添 加 类 别 </a></div> 
<div id="showtype" align="center"><a href="showtype.php" target="mainFrame" id="menu"> 查 看 类 别 </a></div> 
</div> 

<div id="hidediv" align="center"></div> 

<!- 商品 管理 菜单 ”一 > 

<div id="commo" align="center" onclick="javascript:change(two,type);"> 类 别管 理 </div> 

<div id="two"style="display:none"> 

<!-- 商品 管理 子 菜单 -> 


</div> 
SS 人 
反 除了 加 粗 的 这 名 称 和 js 事件 不 同 外 ， 其 他 菜单 的 结构 完全 相同 ， 此 时 只 需 修改 超 链接 即 可 。 


全 s 广 刘 除了 第 一 个 类 别 菜单 的 子 菜单 display 样式 为 空 外 ， 其 他 几 个 子 菜单 的 div 样式 都 为 “display= 


none;”。 


该 页 面 在 Dreamweaver 中 的 效果 如 图 26.24 所 示 。 
因为 其 他 子 菜单 的 样式 为 display=none， 所 以 只 有 “类 别管 


和 
区 吉 法 别 ， 


理 ” 子 菜单 是 可 见 的 ， 下 面 为 它 添加 js 事件 。leftjs 脚本 文件 的 代 
码 如 下 : ee 
(代码 位 置 ， 光盘 \TM\Instance\26\admin\js\left.js) 26.24 div 树 形 菜单 


function change(nu,IxX{ 
if(nu.style.display == "none"){ 
nu.style.display = "™"; 
Ix.style.background="url(images/admin(5).gif)"; 
}else{ 
nu.style.display = "none"; 
Ix.style.background="url(images/admin(1).gif)"; 
} 


时 

最 后 在 left.css 中 设置 div 的 长 、 宽 等 一 些 默 认 参 数 。 一 个 简单 而 又 实用 的 树 形 菜单 就 完成 了 。 对 于 后 
台 的 大 部 分 模块 来 说 ， 其 功能 实现 的 方法 和 开发 步 又 在 前 台 的 模块 设计 中 基本 都 已 经 介绍 过 。 由 于 篇 幅 所 
限 ， 这 里 不 再 对 后 台 管 理 模块 进行 详细 讲解 。 


20:12 小 结 


本 项 目 使 用 Smarty、PDO 数据 库 抽象 层 和 Ajax 等 目前 的 主流 技术 ， 开 发 一 个 电子 商务 平台 。 通 过 本 
章 的 学 习 ， 读 者 不 仅 可 以 了 解 电子 商务 网 站 的 开发 流程 ， 而 且 还 可 以 熟悉 购物 车 、 订 单 及 加 密 技 术 的 开发 
思想 。 
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WI 第 27 章 易 查 供求 信息 网 
Wm 第 28 章 图 书馆 管理 系统 


讶 


EI 


第 L 1 
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( 铝 ! 视频 讲解 : 125 分 钟 ) 


在 全 球 知识 经 济 和 信息 化 高 速 发 展 的 今天 ， 信 息 化 是 决定 企业 成 
败 的 关键 因素 ， 企 业 需 要 在 网 站 上 发 布 供求 信息 ， 以 促使 企业 在 同 领 
域 中 得 到 突飞猛进 的 发 展 。 

一 个 广泛 的 、 快 速 的 、 自 由 的 信息 交流 平台 ， 为 用 户 带 来 方便 的 
同时 ， 也 会 给 企业 带 来 无 限 商机 。 于 是 ， 以 因特网 为 基础 的 信息 交流 
平台 即 易 查 供求 信息 网 出 现 了 。 易 查 供求 信息 网 致力 于 优化 信息 交流 ， 
实现 信息 的 快速 交流 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

M 了 解 使 当前 窗口 承载 框架 页 中 的 起 链接 页 面 

WH 了 解 如 何 自 动 计算 以 系统 日 期 为 基数 的 相对 日 期 

WI 了 解 do…while 循环 语句 的 应 用 

让 了 解 查询 关键 宁 描 红 技 术 

让 了 解 框架 技术 在 Web 网 站 中 的 应 用 
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27.1 易 查 供求 信息 网 概述 


吉林 省 明日 科技 有 限 公 司 是 一 家 以 整合 渠道 资源 为 主 的 高 科技 公司 。 为 了 扩大 企业 规模 ， 增 强 企 业 的 
竞争 力 ， 该 公司 决定 向 多 元 化 发 展 ， 借 助 mtemet 在 国内 的 快速 发 展 ， 聚 集 部 分 资金 投入 网 站 建设 ， 为 企业 
和 用 户 提供 综合 信息 服务 ， 以 向 企业 提供 有 偿 信息 服务 为 莉 利 方式 ， 打 造 一 个 全 新 的 供求 信息 网 。 例 如 ， 
提供 企业 广告 、 发 布 各 类 免费 供求 信息 、 发 布 企业 付费 信息 等 服务 方式 。 现 需要 委托 其 他 单位 开发 一 个 综 
合 信息 网 站 。 


27.2 系统 分 析 


27.2.1 需求 分 析 


对 于 信息 网 站 来 说 ， 用 户 的 访问 量 是 至 关 重 要 的 。 如 果 网 站 的 访问 量 很 低 ， 那 么 就 很 少 有 企业 会 要 求 
为 他 提供 有 偿 服 务 ， 也 就 没有 利润 可 言 了 。 因 此 信息 网 站 必须 为 用 户 提供 大 量 的、 免费 的 、 有 价值 的 信息 
才能 够 吸引 用 户 。 为 此 ， 网 站 不 仅 要 为 企业 提供 各 种 有 偿 服 务 ， 还 需要 额外 为 用 户 提供 大 量 的 无 偿 服务 。 
通过 与 企业 的 实际 接触 和 沟通 ， 确 定 网 站 应 包括 招聘 信息 、 求 职 信息 、 培 训 信 息 、 公 寓 信 息 、 家 教 信息 、 
车 辆 信息 、 物 品 求购 、 物 品 出 售 、 求 竞 出 竞 、 寻 求 合作 、 企 业 广 告 等 服务 。 

通过 实际 调查 ， 要 求 供求 信息 网 应 具有 以 下 功能 。 
界面 设计 美观 大 方 、 方 便 、 快 捷 、 操 作 灵活 ， 树 立 企业 形象 。 
实现 强大 的 供求 信息 查询 ， 支 持 模糊 查询 。 
用 户 不 需要 注册 ， 便 可 免费 发 布 供求 信息 。 
免费 发 布 的 供求 信息 必须 经 后 台 审 核 后 才能 正式 发 布 ， 避 免 不 良 信息 。 
支持 海量 数据 录入 。 
由 于 供求 信息 数据 量 大 ， 后 台 应 该 可 以 随时 清理 数据 。 
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27.2.2 ”可行 性 分 析 


根据 《GB8567 一 88 计算 机 软件 产品 开发 文件 编制 指南 》 中 可 行 性 分 析 的 要 求 ， 制 定 可 行 性 研究 报告 
如 下 。 


1. 引言 

(1) 编写 目的 

为 了 给 企业 的 决策 层 提供 是 否 进 行 项 目 实施 的 参考 依据 ， 现 以 文件 的 形式 分 析 项 目的 风险 、 项 目 需要 
的 投资 与 效益 。 

(2) 背景 


吉林 省 明日 科技 有 限 公司 是 一 家 以 整合 渠道 资源 为 主 的 高 科技 公司 。 企 业 为 了 不 断 满足 客户 的 需求 ， 
为 达到 企业 在 同行 业 领 域 中 的 领先 地 位 ， 现 需要 委托 其 他 公司 开发 一 个 综合 信息 网 ， 项 目 名 称 为 易 查 供求 


信息 网 。 
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2. 可 行 性 研究 的 前 提 

(1) 要 求 

易 查 供求 信息 网 要 求 能 够 提供 信息 搜索 信息 定位 描 红 ， 发 布 免费 信息 ， 发 布 付费 信息 ， 发 布 企业 广告 ， 
对 各 类 发 布 的 信息 进行 审核 、 删 除 、 检 索 等 功能 。 

(2) 目标 

易 查 供求 信息 网 的 主要 目标 是 提供 强大 的 搜索 功能 ， 准 确 的 信息 描 红 定位 功能 ， 付 费 信息 的 管理 、 免 
费 信 息 的 审核 和 删除 功能 。 

(3) 条 件 、 假 定 和 限制 

项 目 需要 在 两 个 月 内 交付 用 户 使 用 。 系 统 分 析 师 需要 3 天 内 到 位 ， 用 户 需 要 4 天 时 间 确 认 需 求 分 析 文 
档 。 去 除 员 工 两 个 月 的 正常 休息 日 ， 程 序 开发 人 员 需 要 在 1 个 月 零 几 天 的 时 间 内 进行 系统 设计 、 程 序 编码 、 
系统 测试 、 程 序 调试 和 网 站 部 署 工作 。 

(4) 评价 尺度 

根据 用 户 的 要 求 ， 系 统 应 以 搜索 引擎 为 主 ， 对 于 发 布 的 供求 信息 应 能 及 时 准确 地 保存 、 审 核 、 查 询 、 
描 红 定 位 。 由 于 用 户 存在 多 个 营业 点 ， 系 统 应 具有 局 域 网 操作 的 能 力 ， 在 多 个 营业 点 同时 运行 系统 时 ， 系 
统 中 各 项 操作 的 延 时 不 能 超过 10 秒 钟 。 此 外 ， 在 系统 出 现 故障 时 ， 应 能 及 时 进行 恢复 。 

3. 投资 及 效益 分 析 

(1) 支出 

根据 系统 的 规模 及 两 个 月 的 项 目 开发 周期 ， 公 司 决 定投 入 5 个 人 人。 因此， 公司 将 直接 支付 8 万 元 的 工 
资 及 各 种 福利 待遇 。 在 项 目 安装 及 调试 阶段 ， 用 户 培训 、 员 工 出 差 等 费用 支出 需要 2 万 元 。 在 项 目 维护 阶 
段 预计 需要 投入 2 万 元 的 资金 ， 累 计 项 目 投 入 需要 12 万 元 资金 。 

(2) 收益 

用 户 提供 项 目 资金 30 万 元 。 对 于 项 目 运行 后 进行 的 改动 , 采取 协商 的 原则 根据 改动 规模 额外 提供 资金 。 
因此 从 投资 与 收益 的 效益 比 上 ， 公 司 可 以 获得 18 万 元 的 利润 。 

项 目 完成 后 ， 将 给 公司 提供 资源 储备 ， 包 括 技术 、 经 验 的 积累 ， 以 后 再 开发 类 似 的 项 目 时 ， 可 以 极 大 
地 缩短 项 目 开 发 周期 。 

4. 结论 


根据 上 面 的 分 析 ， 技 术 上 不 会 存在 问题 ， 因 此 项 目 延期 的 可 能 性 很 小 。 在 效益 上 ， 公 司 投入 5 个 人 、 
两 个 月 的 时 间 获 利 18 万 元 ， 比 较 可 观 。 在 公司 今后 的 发 展 上 可 以 储备 网 站 开发 的 经 验 和 资源 ， 因 此 认为 该 
项 目 可 以 开发 。 


27.3 系统 设计 


27.3.1 系统 目标 


根据 需求 分 析 的 描述 以 及 与 用 户 的 沟通 ， 现 制定 网 站 实现 的 目标 如 下 。 

系统 采用 人 机 对 话 方式 ， 界 面 美观 友好 ， 界 面 简洁 、 框 架 清晰 、 美 观 大 方 。 
灵活 快速 地 填写 供求 信息 ， 使 信息 传递 更 快捷 。 

信息 查询 灵活 、 方 便 ， 数 据 存储 安全 可 靠 。 

实施 强大 的 后 台 审核 功能 。 
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实现 强大 的 搜索 引擎 ， 支 持 模 糊 查 询 、 关 键 字 描 红 功 能 等 。 

对 用 户 输入 的 数据 ， 系 统 进行 严格 的 数据 检验 ， 尽 可 能 排除 人 为 的 错误 。 

网 站 最 大 限度 地 实现 易 维护 性 和 易 操作 性 。 

为 充分 展现 网 站 的 交互 性 ， 供 求 信息 网 采用 动态 网 页 技术 实现 用 户 信息 在 线 发 布 。 
具备 完善 的 后 台 管理 功能 ， 能 够 及 时 、 准 确 地 对 网 站 进行 维护 和 更 新 。 


27.3.2 ”系统 功能 结构 


易 查 供求 信息 网 前 台 功 能 结构 图 如 图 27.1 所 示 。 
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27.1 易 查 供求 信息 网 前 台 功 能 结构 图 
易 查 供求 信息 网 后 台 功 能 结构 图 如 图 27.2 所 示 。 
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图 27.2 吻 查 供求 信息 网 后 台 功 能 结构 图 
易 查 供求 信息 网 的 系统 流程 如 图 27.3 所 示 。 
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图 27.3 系统 流程 图 
27.3.3 ”系统 预览 


易 查 供求 信息 网 由 多 个 程序 页 面 组 成 。 下 面 列 出 几 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 

前 台 首页 如 图 27.4 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 查询 、 企 业 广告 信息 显示 、 后 台 登 录入 口 等 功能 。 
搜索 引擎 页 面 如 图 27.5 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 快速 检索 、 查 询 关 键 字 描 红 等 功能 。 发 布 免费 信 
息 页 面 如 图 27.6 所 示 ， 该 页 面 用 于 实现 发 布 分 类 的 免费 信息 功能 。 付 费 管理 页 面 如 图 27.7 所 示 ， 该 页 面 用 
于 实现 付费 信息 分 类 查看 、 付 费 信息 审核 、 付 费 信息 删除 等 功能 。 
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易 查 供求 信息 网 由 多 个 程序 页 面 组 成 。 下 面 列 出 几 个 典型 页 面 ， 其 他 页 面 参见 光盘 中 的 源 程序 。 

前 台 首页 如 图 27.4 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 查询 、 企 业 广告 信息 显示 、 后 台 登 录入 口 等 功能 。 
搜索 引擎 页 面 如 图 27.5 所 示 ， 该 页 面 用 于 实现 各 类 信息 的 快速 检索 、 查 询 关 键 字 描 红 等 功能 。 发 布 免费 信 
息 页 面 如 图 27.6 所 示 ， 该 页 面 用 于 实现 发 布 分 类 的 免费 信息 功能 。 付 费 管理 页 面 如 图 27.7 所 示 ， 该 页 面 用 
于 实现 付费 信息 分 类 查看 、 付 费 信息 审核 、 付 费 信息 删除 等 功能 。 
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免费 管理 页 面 如 图 27.8 所 示 ， 该 页 面 用 于 实现 免费 信息 分 类 查看 、 免 费 信息 审核 、 免 费 信 息 删除 等 功 
能 。 管 理 员 登 录 页 面 如 图 27.9 所 示 ， 该 页 面 用 于 实现 对 管理 员 登 录 的 用 户 名 和 密码 进行 验证 等 功能 。 
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图 27.8 免费 管理 图 27.9 管理 员 登 录 


27.3.4 文件 夹 组 织 结构 


多 il 视频 讲解 : 光盘 \TM\Video\ 第 27 章 \ 文 件 夹 组 织 结构 .exe 

在 编写 代码 前 ， 可 以 把 系统 中 可 能 用 到 的 文件 夹 先 创建 出 来 (如 创建 一 个 名 为 admin 的 文件 夹 ， 用 于 
保存 网 站 的 后 台 文 件 ) ， 这 样 不 但 可 以 方便 以 后 的 开发 工作 ， 也 可 以 规范 网 站 的 整体 架构 。 笔 者 在 开发 易 
查 供 求 信息 网 时 ， 设 计 了 如 图 27.10 所 示 的 文件 夹 架 构图 。 在 开发 时 ， 只 需要 将 所 创建 的 文件 保存 在 相应 的 


文件 夹 中 即 可 。 


用 于 存储 网 站 后 台 文件 

用 于 存储 网 站 后 台 使 用 的 图 片 资源 
用 于 存储 数据 库 连 接 文件 

用 于 存储 网 站 使 用 的 CSS 样式 表 
用 于 存储 数据 库 文件 

用 于 存储 网 站 前 台 使 用 的 图 片 资源 
用 于 存储 网 站 使 用 的 自 定义 函数 


27.10 文件 夹 组 织 结构 
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27.4 数据 库 设 计 
名 4 视频 讲解 :光盘 \TM\Video\ 第 27 章 \ 数 据 库 设计 .exe 


27.4.1 数据 库 分 析 


本 系统 是 一 个 中 小 型 的 供求 信息 平台 ， 但 是 由 于 平台 会 涉及 到 海量 数据 ， 因 此 需要 充分 考虑 到 成 本 问 
题 及 使 用 需求 (如 跨 平 台 ) 等 问题 , 而 MySQL 是 世界 上 最 为 流行 的 开放 源 代码 的 数据 库 , 是 完全 网 络 化 的 、 
跨 平台 的 关系 型 数据 库 系 统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 


27.4.2 ”数据 库 概念 设计 


根据 前 面 对 系 统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实体 分 别 为 免费 信息 实体 、 
付费 信息 实体 、 广 告 信息 实体 、 管 理 员 实 体 。 下 面 将 介绍 这 几 个 实体 的 E-R 图 。 

1. 免费 信息 实体 

免费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 审 核 状态 和 发 布 时 间 
属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 审核 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 和 否 ”。 免 费 信息 实体 的 
E-R 图 如 图 27.11 所 示 。 

2. 付费 信息 实体 

付费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 发 布 时 间 、 截 止 时 间 
和 审核 状态 属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 付费 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 否 ”。 付 费 
信息 实体 的 E-R 图 如 图 27.12 所 示 。 


图 27.11 免费 信息 实体 E-R 图 图 27.12 ”付费 信息 实体 的 E-R 图 


3. 广告 信息 实体 
广告 信息 实体 包括 编号 、 信 息 标题 、 信 息 内 容 、 发 布 时 间 和 推荐 状态 属性 。 其 中 推荐 状态 属性 用 来 标 
识 信息 是 否 在 前 台 显示 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 和 否 ”。 广 告 信息 实体 的 E-R 图 如 图 27.13 所 示 。 
4. 管理 员 实体 
管理 员 实体 包括 编号 、 管 理 员 名 和 加 密 密 码 属性 。 管 理 员 实体 的 E-R 图 如 图 27.14 所 示 。 
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27.4 数据 库 设 计 
名 4 视频 讲解 :光盘 \TM\Video\ 第 27 章 \ 数 据 库 设计 .exe 


27.4.1 数据 库 分 析 


本 系统 是 一 个 中 小 型 的 供求 信息 平台 ， 但 是 由 于 平台 会 涉及 到 海量 数据 ， 因 此 需要 充分 考虑 到 成 本 问 
题 及 使 用 需求 (如 跨 平 台 ) 等 问题 , 而 MySQL 是 世界 上 最 为 流行 的 开放 源 代码 的 数据 库 , 是 完全 网 络 化 的 、 
跨 平台 的 关系 型 数据 库 系 统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 


27.4.2 ”数据 库 概念 设计 


根据 前 面 对 系 统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实体 分 别 为 免费 信息 实体 、 
付费 信息 实体 、 广 告 信息 实体 、 管 理 员 实 体 。 下 面 将 介绍 这 几 个 实体 的 E-R 图 。 

1. 免费 信息 实体 

免费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 审 核 状态 和 发 布 时 间 
属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 审核 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 和 否 ”。 免 费 信息 实体 的 
E-R 图 如 图 27.11 所 示 。 

2. 付费 信息 实体 

付费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 发 布 时 间 、 截 止 时 间 
和 审核 状态 属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 付费 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 否 ”。 付 费 
信息 实体 的 E-R 图 如 图 27.12 所 示 。 


图 27.11 免费 信息 实体 E-R 图 图 27.12 ”付费 信息 实体 的 E-R 图 


3. 广告 信息 实体 
广告 信息 实体 包括 编号 、 信 息 标题 、 信 息 内 容 、 发 布 时 间 和 推荐 状态 属性 。 其 中 推荐 状态 属性 用 来 标 
识 信息 是 否 在 前 台 显示 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 和 否 ”。 广 告 信息 实体 的 E-R 图 如 图 27.13 所 示 。 
4. 管理 员 实体 
管理 员 实体 包括 编号 、 管 理 员 名 和 加 密 密 码 属性 。 管 理 员 实体 的 E-R 图 如 图 27.14 所 示 。 


@@ 


PHP+MySQL 开发 实战 


27.4 数据 库 设 计 
名 4 视频 讲解 :光盘 \TM\Video\ 第 27 章 \ 数 据 库 设计 .exe 


27.4.1 数据 库 分 析 


本 系统 是 一 个 中 小 型 的 供求 信息 平台 ， 但 是 由 于 平台 会 涉及 到 海量 数据 ， 因 此 需要 充分 考虑 到 成 本 问 
题 及 使 用 需求 (如 跨 平 台 ) 等 问题 , 而 MySQL 是 世界 上 最 为 流行 的 开放 源 代码 的 数据 库 , 是 完全 网 络 化 的 、 
跨 平台 的 关系 型 数据 库 系 统 ， 这 正好 满足 了 中 小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 


27.4.2 ”数据 库 概念 设计 


根据 前 面 对 系 统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实体 分 别 为 免费 信息 实体 、 
付费 信息 实体 、 广 告 信息 实体 、 管 理 员 实 体 。 下 面 将 介绍 这 几 个 实体 的 E-R 图 。 

1. 免费 信息 实体 

免费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 审 核 状态 和 发 布 时 间 
属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 审核 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 和 否 ”。 免 费 信息 实体 的 
E-R 图 如 图 27.11 所 示 。 

2. 付费 信息 实体 

付费 信息 实体 包括 编号 、 信 息 类 型 、 信 息 标题 、 信 息 内 容 、 联 系 人 、 联 系 电话 、 发 布 时 间 、 截 止 时 间 
和 审核 状态 属性 。 其 中 审核 状态 属性 用 来 标识 信息 是 否 付费 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 否 ”。 付 费 
信息 实体 的 E-R 图 如 图 27.12 所 示 。 


图 27.11 免费 信息 实体 E-R 图 图 27.12 ”付费 信息 实体 的 E-R 图 


3. 广告 信息 实体 
广告 信息 实体 包括 编号 、 信 息 标题 、 信 息 内 容 、 发 布 时 间 和 推荐 状态 属性 。 其 中 推荐 状态 属性 用 来 标 
识 信息 是 否 在 前 台 显示 ，“1” 表 示 “ 是 ”，“0” 表 示 “ 和 否 ”。 广 告 信息 实体 的 E-R 图 如 图 27.13 所 示 。 
4. 管理 员 实体 
管理 员 实体 包括 编号 、 管 理 员 名 和 加 密 密 码 属性 。 管 理 员 实体 的 E-R 图 如 图 27.14 所 示 。 
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图 27.13 广告 信息 实体 E-R 27.14 管理 员 实 体 E-R 


27.4.3 创建 数据 库 及 数据 表 


结合 实际 情况 及 对 用 户 需求 的 分 析 ， 可 知 易 查 供求 信息 网 OO 
中 应 用 的 db_pursey 数据 库 主 要 包含 如 下 4 个 数据 表 , 如 图 27.15 。 | waamm MisA pb2312_rninese al。 符 开 RN 


所 示 tb aduertising MHSAW ob2312_chinese_el 。 全 业 广 尖 人 息 表 

bn MiEAM 。 gb2312_cninese_rl 。 免费 慌 信 息 表 
各 数据 表 的 表 结 构 如 图 27.16 一 图 27.19 所 示 。 全 Jeaguerinfo MyISAM 9b2312_chinese_ti 。 付费 人 求 信息 表 
1. tb_admin 〈 管 理 员 信息 表 ) 图 27.15 易 查 供求 信息 网 数据 表 


管理 员 信 息 表 主 要 用 于 存储 管理 员 的 信息 。 该 数据 表 的 结构 如 图 27.16 所 示 。 
2. tb_advertising (企业 广告 信息 表 ) 
企业 广告 信息 表 主 要 用 于 存储 企业 发 布 的 广告 信息 。 该 数据 表 的 结构 如 图 27.17 所 示 。 


六 服务 器 : localhost } 冰 数据 库 : db_pursey》 川 表 : tb_advertlsing 
FE 天 半 理 Nal 于 认可 外 如 明 


地 Ht 再 uio_rerement 。 自动 号 
说 四 te varcnarit gb2312_shinese ol 再 

| 友和 标 理 autp_ntrement Ez 编号 Comemt -varcmanson) 902312_chinese_o 再 

ate 

fog 


四 服务 器 : localhost 各 数据 库 : db_pursey 图 表 :由 -admin 
宁 有 & 。 类型 可 理 Nul 里 认 要 外 


mame vartharis0) ob2312_chinese_el 是 NULL 管理 只 名 aietme 
pwd varchar(50) gb2312_chinese_tl 是 NULL 管理 员 衬 码 nt 


图 27.16 管理 员 信息 表 结构 图 27.17 企业 广告 信息 表 结构 
3. tb_info (免费 供求 信息 表 ) 
免费 供求 信息 表 主 要 用 于 存储 用 户 免费 发 布 的 供求 信息 。 该 数据 表 的 结构 如 图 27.18 所 示 。 
4. tb_leaguerinfo 〈 付 费 供 求 信息 表 ) 
付费 供求 信息 表 主 要 用 于 存储 付费 的 供求 信息 。 该 数据 表 的 结构 如 图 27.19 所 示 。 
园 服务 器 : localhost ， 四 数据 库 : db_pursey ， 莉 表 :tb_leaguerinfo 
他 恨 天 型 整理 Ml 黑人 鲁 外 说明 
nik) 再 auto_inerement 自动 
inttd) varenari20) 92312_chinese_ol 否 信息 关 开 


vareharlan) 的 2312_rhinase_ri varchansD) go2312_chinese_ci 否 信息 标题 
Maiehari50) 的 2312_rhinase_ri Tcharl500) go2312_chinese_d 否 信息 内 容 


vateharisn0) oh2312_chinese_cl warcnart20) gh2312_chinesa_ol 百 联系 人 
vatehar20) gh2312_chinese_cf varcnarl30) ”go2312_chinesa_ol 百 联系 虹 话 


vateharlan) 的 2312_chinese_ef date 加 发 市 E 央 
int) date 至 弄 旧 期 
atelime nt) 否 市 护 帮 
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后 台 登 录入 口 : 为 管理 员 进 入 后 台 提供 一 个 入 口 。 
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第 27 章 易 查 供求 信息 网 


各 区 域 的 介绍 及 所 对 应 的 PHP 文件 如 表 27.1 所 示 。 
表 27.1 页 面 框架 中 各 区 域 介绍 及 对 应 PHP 文件 


对 应 PHP 文件 


导航 栏 提供 查看 各 类 信息 的 超 链接 top php 

信息 检索 区 。 | ”企业 广告 显示 及 各 类 信息 检索 left php 

内 容 显 示 区 | 。 根据 用 户 请 求 显示 相应 内 容 根据 请 求 加 载 相应 的 PHP 文件， 默认 加 载 main php 
版 权 区 显示 版 权 信 息 bottom php 


27.5.2” 超 链接 技术 


胃 链 接 在 本 质 上 属于 一 个 网 页 的 一 部 分 ， 它 是 一 种 允许 用 户 同 其 他 网 页 或 站 点 之 间 进 行 连 接 的 元 素 。 
各 个 网 页 链接 在 一 起 后 ， 才 能 真正 构成 一 个 网 站 。 按 照 使 用 对 象 的 不 同 ， 网 页 中 的 链接 可 以 分 为 文本 超 链 
接 、 图 像 超 链接 、E-mail 链接 、 锚 点 链接 、 多 媒体 文件 链接 和 空 链接 等 。 

图 像 不 但 可 以 建立 超 链 接 ， 还 可 以 实现 图 像 映射 。 图 像 映射 是 指 一 幅 图 像 可 以 建立 多 个 超 链接 ， 即 在 
图 像 上 定义 多 个 区 域 ， 每 个 区 域 链接 到 不 同 的 地 址 ， 这 样 的 区 域 称 为 热 区 〈 也 称 为 热点 ，Hotspot) 。 本 项 
目 就 是 应 用 图 像 热 区 实现 的 功能 导航 。 

像 映 射 有 两 种 : 服务 器 端 映射 〈Server-side Image Map) 和 客户 端 映射 (Client-side Image Map) 。 目 
前 使 用 最 多 的 是 客户 端 映射 ， 因 为 客户 端 映射 使 图 像 上 对 应 区 域 的 坐标 以 及 链接 的 URL 地 址 都 在 浏览 器 端 
读 入 ， 省 去 和 服务 器 之 间 互 传 坐标 和 URL 的 时 间 。 

在 PHP 文件 中 ， 使 用 <MAP> 标 记 创建 图 像 映射 。 语 法 如 下 : 

<IMG SRC="file_name" USEMAP="#MapName"> 

<MAP NAME="MapName"> 

<AREA SHAPE="value" COORDS=" 坐 标 "HREF="URL" ALT=" 描 述 文字 "> 

<AREA SHAPE="value" COORDS=" 坐 标 "HREF="URL" ALT=" 描 述 文字 "> 


pa 


</MAP> 
在 <IMG> 标 记 中 设置 属性 USEMAP， 确 定 创建 图 像 映射 。<MAP> 标 记 的 属性 如 表 27.2 所 示 。 


表 27.2 <MAP> 标 记 的 属性 


<MAP> 标 记 的 属性 描述 
NAME 图 像 映射 的 名 称 
SHAPE 定义 图 像 映射 区 域 的 名 称 
COORDS 设 定 区 域 坐 标 
HREF 设 定 区 域 的 链接 地 址 
AILT 设 定 区 域 链接 的 描述 文字 


在 <MAP> 标 记 中 ， 根 据 属 性 SHAPE 的 取 值 不 同 ， 相 应 坐标 的 设 定 也 不 同 。 下 面 介 绍 属性 SHAPE 的 3 
种 取 值 以 及 相应 坐标 的 设 定 。 
设 定 属性 SHAPE 的 属性 值 为 rect。 属 性 SHAPE 取 值 为 rect， 表 示 和 矩形 区 域 ， 属 性 COORDS 的 坐 
标 形式 为 “X1,Y1,X2,Y2”。 其 中 ，X1、Y1 代表 矩形 左上 角 的 X、Y 坐标 ，X2、Y2 代表 和 矩形 右 
下 角 的 义 、Y 坐标。 
设 定 属性 SHAPE 的 属性 值 为 circle。 属 性 SHAPE 取 值 为 circle， 表 示 圆 形 区 域 ， 属 性 COORDS 
的 坐标 形式 为 “X.Y:r”。 其 中 ，X、Y 为 圆心 坐标 ，r 为 圆 的 半径 。 
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各 区 域 的 介绍 及 所 对 应 的 PHP 文件 如 表 27.1 所 示 。 
表 27.1 页 面 框架 中 各 区 域 介绍 及 对 应 PHP 文件 
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</MAP> 
在 <IMG> 标 记 中 设置 属性 USEMAP， 确 定 创建 图 像 映射 。<MAP> 标 记 的 属性 如 表 27.2 所 示 。 
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在 <MAP> 标 记 中 ， 根 据 属 性 SHAPE 的 取 值 不 同 ， 相 应 坐标 的 设 定 也 不 同 。 下 面 介 绍 属性 SHAPE 的 3 
种 取 值 以 及 相应 坐标 的 设 定 。 
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各 区 域 的 介绍 及 所 对 应 的 PHP 文件 如 表 27.1 所 示 。 
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内 容 显 示 区 | 。 根据 用 户 请 求 显示 相应 内 容 根据 请 求 加 载 相应 的 PHP 文件， 默认 加 载 main php 
版 权 区 显示 版 权 信 息 bottom php 


27.5.2” 超 链接 技术 


胃 链 接 在 本 质 上 属于 一 个 网 页 的 一 部 分 ， 它 是 一 种 允许 用 户 同 其 他 网 页 或 站 点 之 间 进 行 连 接 的 元 素 。 
各 个 网 页 链接 在 一 起 后 ， 才 能 真正 构成 一 个 网 站 。 按 照 使 用 对 象 的 不 同 ， 网 页 中 的 链接 可 以 分 为 文本 超 链 
接 、 图 像 超 链接 、E-mail 链接 、 锚 点 链接 、 多 媒体 文件 链接 和 空 链接 等 。 

图 像 不 但 可 以 建立 超 链 接 ， 还 可 以 实现 图 像 映射 。 图 像 映射 是 指 一 幅 图 像 可 以 建立 多 个 超 链接 ， 即 在 
图 像 上 定义 多 个 区 域 ， 每 个 区 域 链接 到 不 同 的 地 址 ， 这 样 的 区 域 称 为 热 区 〈 也 称 为 热点 ，Hotspot) 。 本 项 
目 就 是 应 用 图 像 热 区 实现 的 功能 导航 。 

像 映 射 有 两 种 : 服务 器 端 映射 〈Server-side Image Map) 和 客户 端 映射 (Client-side Image Map) 。 目 
前 使 用 最 多 的 是 客户 端 映射 ， 因 为 客户 端 映射 使 图 像 上 对 应 区 域 的 坐标 以 及 链接 的 URL 地 址 都 在 浏览 器 端 
读 入 ， 省 去 和 服务 器 之 间 互 传 坐标 和 URL 的 时 间 。 

在 PHP 文件 中 ， 使 用 <MAP> 标 记 创建 图 像 映射 。 语 法 如 下 : 

<IMG SRC="file_name" USEMAP="#MapName"> 

<MAP NAME="MapName"> 

<AREA SHAPE="value" COORDS=" 坐 标 "HREF="URL" ALT=" 描 述 文字 "> 

<AREA SHAPE="value" COORDS=" 坐 标 "HREF="URL" ALT=" 描 述 文字 "> 


pa 


</MAP> 
在 <IMG> 标 记 中 设置 属性 USEMAP， 确 定 创建 图 像 映射 。<MAP> 标 记 的 属性 如 表 27.2 所 示 。 


表 27.2 <MAP> 标 记 的 属性 


<MAP> 标 记 的 属性 描述 
NAME 图 像 映射 的 名 称 
SHAPE 定义 图 像 映射 区 域 的 名 称 
COORDS 设 定 区 域 坐 标 
HREF 设 定 区 域 的 链接 地 址 
AILT 设 定 区 域 链接 的 描述 文字 


在 <MAP> 标 记 中 ， 根 据 属 性 SHAPE 的 取 值 不 同 ， 相 应 坐标 的 设 定 也 不 同 。 下 面 介 绍 属性 SHAPE 的 3 
种 取 值 以 及 相应 坐标 的 设 定 。 
设 定 属性 SHAPE 的 属性 值 为 rect。 属 性 SHAPE 取 值 为 rect， 表 示 和 矩形 区 域 ， 属 性 COORDS 的 坐 
标 形式 为 “X1,Y1,X2,Y2”。 其 中 ，X1、Y1 代表 矩形 左上 角 的 X、Y 坐标 ，X2、Y2 代表 和 矩形 右 
下 角 的 义 、Y 坐标。 
设 定 属性 SHAPE 的 属性 值 为 circle。 属 性 SHAPE 取 值 为 circle， 表 示 圆 形 区 域 ， 属 性 COORDS 
的 坐标 形式 为 “X.Y:r”。 其 中 ，X、Y 为 圆心 坐标 ，r 为 圆 的 半径 。 
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回 ” 设 定 属性 SHAPE 的 属性 值 为 poly。 属 性 SHAPE 取 值 为 poly， 表 示 多 边 形 区 域 ， 属 性 COORDS 
的 坐标 形式 为 “X1.Y1.X2.Y2,…”。 其 中 ，Xn、Yma 代表 构成 多 边 形 每 一 点 的 坐标 值 ，n 的 取 值 为 
1，2，3，…。 多 边 形 有 几 边 就 有 几 对 X、 立 坐标 。 
下 面 在 Dreamweaver 编辑 器 中 创建 图 像 映射 ， 单 击 图 像 不 同 区 域 ， 链 接 到 不 同 的 文件 。 创 建 图 像 映射 
的 步骤 如 下 : 
(1) 在 Dreamweaver 编辑 器 中 选中 所 要 编辑 的 图 像 ， 在 属性 窗口 中 ， 有 3 种 形状 的 热点 工具 ， 分 别 为 
矩形 热点 工具 、 圆 形 热点 i 
选择 一 种 热点 工具 ， 在 图 像 上 按 住 鼠 标 左 键 拖 动 ， 确 定 所 选区 域 。 
(3) 设置 区 域 属性 ， 在 “链接 ”文本 框 中 输入 所 要 链接 书签 的 名 称 ， 在 “ 蔡 代 ” 文 本 框 中 输入 区 
接 文件 ， 如 index.php。 
(4) 重复 步骤 (2) 、 (3) ， 在 图 像 上 定义 不 同 的 区 域 链接 。 


Xx 
荡 
格 


27.5.3 ”前台 首页 的 实现 过 程 


本 系统 中 所 有 的 前 台 页 面 都 采用 了 二 分 栏 结 构 ， 分 为 导航 栏 、 信 息 检索 区 、 内 容 显示 区 和 版 权 区 4 个 
区 域 。 为 了 方便 网 站 的 日 后 维护 ， 将 这 4 个 区 域 形成 单独 的 PHP 文件 ， 然 后 应 用 include 语句 将 这 4 个 文件 
包含 进来 。 前 台 首页 文件 的 代码 如 下 : 

《代码 位 置 ， 光盘 \TMNVInstance\27\index-php) 
<table width="780" border="0" align="center" cellpadding="0" cellspacing="0"> 
<tr valign="top"> 
<td colspan="2"><?php include("top.php");?></td> <!-- 包含 导航 文件 --> 
</tr> 


<tr> 
<!- 包含 信息 检索 文件 --> 
<td width="217" valign="top" background="Images/line2.gif’><?php include("left.php");?></td> 
<!- 包含 内 容 显 示 文件 --> 
<td width="586" valign="top" bgcolor="#FEFEF6"><?php include("main.php");?></td> 
</tr> 
<tr> 
<td colspan="2"><?php include("bottom.php");?></td> <!-- 包含 版 权 信息 文件 --> 
</tr> 
</table> 
其 中 ， 导 航 文件 top.php 页 应 用 了 图 像 映射 来 创建 文件 的 超 链接 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\InstanceW27\top.php) 
<table width="780" height="201" border="0" align="center" cellpadding="0" cellspacing="0"> 


<tr> 
<td height="38"><img src="Images/banner.gif" width="780" height="202" border="0" usemap="#Map"></td> 
</tr> 
</table> 
Es 图 像 映射 = 
<map name="Map"> 
< 联系 我 们 = 
<area shape="rect" coords="685,8,744,27" href="mailto: ee com"> 
E> 设 为 首页 


<area shape="rect" coords="6163,9,666,27" 
‘onClick="this.style.behavior="url(#default#homepage)';this.setHomePage('http://localhost/99pursey/index.php’); 
> 
< 加 入 收藏 3 

<area shape="rect" coords="535,8,593,26" 
href="javascript:window.external.AddFavorite('http://localhost/99pursey/index.php',' 易 查 供求 信息 网 ')"> 
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<area shape="rect" coords="448,8,516,26" href="admin/login.php"> <!-- 后 台 登 录 页 --> 
<area shape="rect" coords="356,8,424,28" href="release.php"> <!-- 发 布 信息 页 --> 
<area shape="rect" coords="680,61,753,82" href="search.php"> <!-- 寻 人 / 物 启示 信息 页 -> 
<area shape="rect" coords="579,62,637,82" href="sale.php"> <!-- 出 售 信息 页 --> 
<area shape="rect" coords="483,62,536,83" href="car.php"> <!- 车 辆 信息 页 --> 
<area shape="rect" coords="385,62,441,83" href="teaching.php"> <!- 家 教 信息 页 --> 
<area shape="rect" coords="290,61,344,83" href="seekjob.php"> <!- 求职 信息 页 --> 
<area shape="rect" coords="678,38,740,58" href="recruitbusiness.php"> ”<!-- 招商 引资 信息 页 --> 
<area shape="rect" coords="579,38,635,58" href="seekbuy.php"> <!- 求购 信息 页 --> 
<area shape="rect" coords="482,38,541,58" href="house.php"> <!-- 房屋 信息 页 --> 
<area shape="rect" coords="381,39,440,60" href= "foster.php"> <!-- 培训 信息 页 --> 
<area shape="rect" coords="290,39,345,59" href="invitejob.php"> <!-- 招聘 信息 页 --> 
<area shape="rect" coords="204,39,263,86" href="index.php"> <!-- 首页 -> 

</map> 


27.6 免费 供求 信息 发 布 模块 设计 
名 m 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 免 费 供 求 信息 发 布 模块 设计 .exe 


27.6.1 免费 供求 信息 发 布 模块 概述 


免费 供求 信息 发 布 功能 是 供求 信息 网 站 最 基本 、 最 核心 的 功能 。 

免费 供求 信息 发 布 模块 可 以 完成 11 种 不 同类 别 信息 的 发 布 。 用 户 可 以 根据 自身 需要 将 供求 信息 发 布 到 
相应 的 信息 类 别 中 《〈 共 包括 11 个 信息 类 别 : 公寓 信息 、 招 聘 信 息 、 求 职 信息 、 培 训 信 息 、 家 教 信息 、 房 尾 
信息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启示 ) 。 信 息 发 布 成 功 后 ， 需 要 管理 员 进行 审 
核 , 只 有 审核 成 功 的 信息 才能 显示 在 前 台 相 应 的 信息 类 别 网 页 中 。 免费 供求 信息 发 布 的 流程 如 图 27.21 所 示 。 


图 27.21 免费 供求 信息 发 布 流程 图 
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<area shape="rect" coords="448,8,516,26" href="admin/login.php"> <!-- 后 台 登 录 页 --> 
<area shape="rect" coords="356,8,424,28" href="release.php"> <!-- 发 布 信息 页 --> 
<area shape="rect" coords="680,61,753,82" href="search.php"> <!-- 寻 人 / 物 启示 信息 页 -> 
<area shape="rect" coords="579,62,637,82" href="sale.php"> <!-- 出 售 信息 页 --> 
<area shape="rect" coords="483,62,536,83" href="car.php"> <!- 车 辆 信息 页 --> 
<area shape="rect" coords="385,62,441,83" href="teaching.php"> <!- 家 教 信息 页 --> 
<area shape="rect" coords="290,61,344,83" href="seekjob.php"> <!- 求职 信息 页 --> 
<area shape="rect" coords="678,38,740,58" href="recruitbusiness.php"> ”<!-- 招商 引资 信息 页 --> 
<area shape="rect" coords="579,38,635,58" href="seekbuy.php"> <!- 求购 信息 页 --> 
<area shape="rect" coords="482,38,541,58" href="house.php"> <!-- 房屋 信息 页 --> 
<area shape="rect" coords="381,39,440,60" href= "foster.php"> <!-- 培训 信息 页 --> 
<area shape="rect" coords="290,39,345,59" href="invitejob.php"> <!-- 招聘 信息 页 --> 
<area shape="rect" coords="204,39,263,86" href="index.php"> <!-- 首页 -> 

</map> 


27.6 免费 供求 信息 发 布 模块 设计 
名 m 视频 讲解 : 光盘 \TM\Video\ 第 26 章 \ 免 费 供 求 信息 发 布 模块 设计 .exe 


27.6.1 免费 供求 信息 发 布 模块 概述 


免费 供求 信息 发 布 功能 是 供求 信息 网 站 最 基本 、 最 核心 的 功能 。 

免费 供求 信息 发 布 模块 可 以 完成 11 种 不 同类 别 信息 的 发 布 。 用 户 可 以 根据 自身 需要 将 供求 信息 发 布 到 
相应 的 信息 类 别 中 《〈 共 包括 11 个 信息 类 别 : 公寓 信息 、 招 聘 信 息 、 求 职 信息 、 培 训 信 息 、 家 教 信息 、 房 尾 
信息 、 车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引 资 、 寻 人 / 物 启示 ) 。 信 息 发 布 成 功 后 ， 需 要 管理 员 进行 审 
核 , 只 有 审核 成 功 的 信息 才能 显示 在 前 台 相 应 的 信息 类 别 网 页 中 。 免费 供求 信息 发 布 的 流程 如 图 27.21 所 示 。 


图 27.21 免费 供求 信息 发 布 流程 图 
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27.6.2 MySQL 数据库 连接 技术 


本 模块 实现 免费 供求 信息 发 布 功 能 主要 应 用 到 如 下 几 个 函数 ， 下 面 进 行 详细 介绍 。 
1. mysql_connect() 函 数 
打开 一 个 到 MySQL 服务 器 的 连接 。 如 果 成 功 则 返回 一 个 MySQL 连接 标识 ， 失 败 则 返回 false。 语 法 
如 下 : 
resource mysql_connect ( [string server [, string username [, string password [, bool new_link [, int client_flags]]]]] ) 
mysql_connect0 函 数 的 参数 说 明 如 表 27.3 所 示 。 
表 27.3 mysql_connect() 函 数 的 参数 说 明 
参数 说 明 
MySQL 服务 器 。 可 以 包括 端口 号 ， 如 "hostname:port"， 或 者 到 本 地 套 接 字 的 路 径 ， 如 对 于 localhost 的 
SE "/pathyto/socket"。 如 果 PHP 指令 mysql.default_host 未 定义 默认 情况 ) ， 则 默认 值 是 'localhost:3306' 
username ”| 用 户 名 。 默 认 值 是 服务 器 进程 所 有 者 的 用 户 名 
assword “| 密码 。 默 认 值 是 空 密码 
如 果 用 同样 的 参数 第 二 次 调用 mysql connectO 函 数 ， 将 不 会 建立 新 连接 ， 而 将 返回 已 经 打开 的 连接 标 
new_link 识 。 参 数 new_link 改变 此 行为 并 使 mysql_connect0 函 数 总 是 打开 新 的 连接 ， 甚 至 mysql_connect0 函 数 
曾 在 前 面 被 同样 的 参数 调用 过 
client_flags 参数 可 以 是 以 下 常量 的 组 合 : MYSQL _CLIENT_SSL、MYSQL _CLIENT_COMPRESS、 
MYSQL CLIENT IGNORE SPACE 或 MYSQL CLIENT INTERACTIVE 
2. mysql_select_db() 函 数 
选择 MySQL 数据 库 。 如 果 成 功 返 回 tue， 失 败 返回 false。 语 法 如 下 : 
bool mysql_select_db ( string database_name [, resource link_identifier] ) 
mysql_select_db() 函 数 设 定 与 指定 的 连接 标识 符 所 关联 的 服务 器 上 的 当前 激活 数据 库 。 如 果 没 有 指定 连 
接 标 识 符 ， 则 使 用 上 一 个 打开 的 连接 。 如 果 没 有 打开 连接 ， 本 函数 将 无 参数 调用 mysql_connect0 函 数 来 尝试 
打开 一 个 并 使 用 。 每 个 其 后 的 mysql_query0 函 数 调用 都 会 作用 于 活动 数据 库 。 
下 面 应 用 mysql_connectO 函 数 连接 MySQL 服务 器 ， 然 后 应 用 mysql_select_db0 函 数 连接 MySQL 数据 


client_flags 


库 。 其 代码 如 下 : 
<?php 
$conn = mysql_connect("localhost", "root", "111"); /连接 MySQL 服务 器 
$db=mysql_select_db("db_pursey", $conn) or die (" 数 据 库 连接 失败 : " .mysql_error()); /连接 数据 库 db _pursey 
mysql_query("set names gb2312"); /采用 GB2312 编码 方式 
?> 


3. mysql_query() 函 数 


mysql_queryO 函 数 用 来 根据 连接 标识 符 向 该 数据 库 服务 器 的 当前 数据 库 发 送 查 询 。 语 法 格式 如 下 : 
int mysql_query(string query ,int [link_identifier]); 


其 中 ，query 是 查询 字符 串 ，link_identifier 是 数据 库 连接 标识 符 。 
mysql_query0 函 数 在 执行 成 功 时 返回 一 个 结果 标识 符 ， 失 败 则 返回 false。 
4. date() 函 数 


date0 函 数 主 要 用 于 格式 化 一 个 本 地 时 间 / 日 期 。 语 法 如 下 : 
string date ( string format , int timestamp) 


@_ 
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27.6.2 MySQL 数据库 连接 技术 


本 模块 实现 免费 供求 信息 发 布 功 能 主要 应 用 到 如 下 几 个 函数 ， 下 面 进 行 详细 介绍 。 
1. mysql_connect() 函 数 
打开 一 个 到 MySQL 服务 器 的 连接 。 如 果 成 功 则 返回 一 个 MySQL 连接 标识 ， 失 败 则 返回 false。 语 法 
如 下 : 
resource mysql_connect ( [string server [, string username [, string password [, bool new_link [, int client_flags]]]]] ) 
mysql_connect0 函 数 的 参数 说 明 如 表 27.3 所 示 。 
表 27.3 mysql_connect() 函 数 的 参数 说 明 
参数 说 明 
MySQL 服务 器 。 可 以 包括 端口 号 ， 如 "hostname:port"， 或 者 到 本 地 套 接 字 的 路 径 ， 如 对 于 localhost 的 
SE "/pathyto/socket"。 如 果 PHP 指令 mysql.default_host 未 定义 默认 情况 ) ， 则 默认 值 是 'localhost:3306' 
username ”| 用 户 名 。 默 认 值 是 服务 器 进程 所 有 者 的 用 户 名 
assword “| 密码 。 默 认 值 是 空 密码 
如 果 用 同样 的 参数 第 二 次 调用 mysql connectO 函 数 ， 将 不 会 建立 新 连接 ， 而 将 返回 已 经 打开 的 连接 标 
new_link 识 。 参 数 new_link 改变 此 行为 并 使 mysql_connect0 函 数 总 是 打开 新 的 连接 ， 甚 至 mysql_connect0 函 数 
曾 在 前 面 被 同样 的 参数 调用 过 
client_flags 参数 可 以 是 以 下 常量 的 组 合 : MYSQL _CLIENT_SSL、MYSQL _CLIENT_COMPRESS、 
MYSQL CLIENT IGNORE SPACE 或 MYSQL CLIENT INTERACTIVE 
2. mysql_select_db() 函 数 
选择 MySQL 数据 库 。 如 果 成 功 返 回 tue， 失 败 返回 false。 语 法 如 下 : 
bool mysql_select_db ( string database_name [, resource link_identifier] ) 
mysql_select_db() 函 数 设 定 与 指定 的 连接 标识 符 所 关联 的 服务 器 上 的 当前 激活 数据 库 。 如 果 没 有 指定 连 
接 标 识 符 ， 则 使 用 上 一 个 打开 的 连接 。 如 果 没 有 打开 连接 ， 本 函数 将 无 参数 调用 mysql_connect0 函 数 来 尝试 
打开 一 个 并 使 用 。 每 个 其 后 的 mysql_query0 函 数 调用 都 会 作用 于 活动 数据 库 。 
下 面 应 用 mysql_connectO 函 数 连接 MySQL 服务 器 ， 然 后 应 用 mysql_select_db0 函 数 连接 MySQL 数据 


client_flags 


库 。 其 代码 如 下 : 
<?php 
$conn = mysql_connect("localhost", "root", "111"); /连接 MySQL 服务 器 
$db=mysql_select_db("db_pursey", $conn) or die (" 数 据 库 连接 失败 : " .mysql_error()); /连接 数据 库 db _pursey 
mysql_query("set names gb2312"); /采用 GB2312 编码 方式 
?> 


3. mysql_query() 函 数 


mysql_queryO 函 数 用 来 根据 连接 标识 符 向 该 数据 库 服务 器 的 当前 数据 库 发 送 查 询 。 语 法 格式 如 下 : 
int mysql_query(string query ,int [link_identifier]); 


其 中 ，query 是 查询 字符 串 ，link_identifier 是 数据 库 连接 标识 符 。 
mysql_query0 函 数 在 执行 成 功 时 返回 一 个 结果 标识 符 ， 失 败 则 返回 false。 
4. date() 函 数 


date0 函 数 主 要 用 于 格式 化 一 个 本 地 时 间 / 日 期 。 语 法 如 下 : 
string date ( string format , int timestamp) 


@_ 
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该 函数 返回 将 参数 timestamp 按照 指定 格式 格式 化 而 产生 的 字符 串 ， 其 中 参数 timestamp 是 可 选 的 ， 默 
认 值 为 tme0， 即 如 果 没有 给 出 时 间 截 则 使 用 本 地 当前 时 间 。 
date0) 函 数 的 参数 format 的 格式 化 选项 如 表 27.4 所 示 。 


加 


表 27.4 参数 format 的 格式 化 选项 
说 明 


小 写 的 上 午 和 下 午 值 ， 返 回 值 为 am 或 pm 


大 写 的 上 午 和 下 午 值 ， 返 回 值 为 AM 或 PM 


Swatch Intemet 标准 时 ， 返 回 值 为 000 一 999 


月 份 中 的 第 几 天 ， 有 前 导 零 的 2 位 数字 ， 返 回 值 为 01 一 31 


星期 中 的 第 几 天 ， 文 本 格式 ，3 个 字母 ， 返 回 值 为 Mon 一 Sun 


月 份 ， 完 整 的 文本 格式 ， 返 回 值 为 January 一 December 


小 时 ，12 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 


-Iam lem > |» 


小 时 ，24 小 时 格式 ， 没 有 前 导 零 ， 返 回 值 为 0 一 23 
小 时 ，12 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 01 一 12 
小 时 ，24 小 时 格式 ， 有 前 导 零 ， 返 回 值 为 00 一 23 

有 前 导 零 的 分 钟 数 ， 返 回 值 为 00 一 59 
判断 是 否 为 夏令 时 ， 如 果 是 夏令 时 返回 值 为 1， 否 则 为 0 


i | 月 份 中 的 第 几 天 ， 没 有 前 导 堆 , 返 加 值 Mt< | 
1 | 星期 数 ， 完 整 的 文本 格式 ， 返 回 值 为 Sunday 一 Satuda 


L 


wl l= lo ls | |B 


判断 是 否 为 半年 ， 如 果 是 闲 年 返回 值 为 1， 否则 为 0 

数字 表示 的 月 份 ， 有 前 导 零 ， 返 回 值 为 01 一 12 

3 个 字母 缩写 表示 的 月 份 ， 返 回 值 为 Jan 一 Dec 

数字 表示 的 月 份 ， 没 有 前 导 零 ， 返 回 值 为 1 一 12 

与 格林 威 治 时 间 相差 的 小 时 数 ， 如 0200 

RFC 822 格式 的 日 期 如 Thu，21 Dec 2000 16:01:07 +0200 
秒 数 ， 有 前 导 零 ， 返 回 值 为 00 一 59 


每 月 天 数 后 面 的 英文 后 级 ，2 个 字符 ， 如 st、nd、rd 或 者 也 。 可 以 和 j 一 起 使 用 


指定 月 份 所 应 有 的 天 数 
本 机 所 在 的 时 区 
从 Unix 纪元 〈January 1 1970 00:00:00 GMT) 开始 至 今 的 秒 数 


星期 中 的 第 几 天 ， 数 字 表 示 ， 返 回 值 为 0 一 6 


ISO-8601 格式 年 份 中 的 第 几 周 ， 每 周 从 星期 一 开始 


4 位 数字 完整 表示 的 年 份 ， 返 回 值 如 1998、2008 


2 位 数字 表示 的 年 份 ， 返 回 值 如 88 或 08 


nk|<|als c=-|- 


N 


年 份 中 的 第 几 天 ， 返 回 值 为 0~~366 
时 差 偏 移 量 的 秒 数 。UTC 西边 的 时 区 偏 移 量 总 是 负 的 ，UTC 东边 的 时 区 偏 移 量 总 是 正 的 ， 返 回 值 为 
-43200 一 43200 


人 os 注 六 有 效 的 时 间 蕉 典型 范围 是 格林 威 治 时 间 1901 年 12 月 13 日 20:45:54 到 2038 年 1 月 19 日 
03:14:07 ( 此 范围 符合 32 位 有 符号 整数 的 最 小 值 和 最 大 值 )。 在 Windows 系统 中 此 范围 限制 为 从 1970 年 
1 月 1 日 到 2038 年 1 月 19 日 。 


_ 回 ) 
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例如 ， 应 用 date0 函 数 格 式 化 一 个 日 期 ， 并 输出 日 期 的 值 。 其 代码 如 下 : 
<?php 

echo date("y:m:d"); 

?> 


结果 为 : 07:07:11。 
27.6.3 ”免费 供求 信息 发 布 模块 的 实现 过 程 


用 户 通过 单 击 前 台 页 面 导航 栏 的 “我 要 发 布 ” 超 链接 ， 进 入 信息 发 布 页 面 ， 如 图 27.22 所 示 。 程 序 会 先 


验证 用 户 输入 的 信息 ， 若 验证 失败 ， 则 返回 信息 发 布 页 面 ， 进 行 相应 提示 。 若 验证 成 功 ， 则 向 数据 库 中 插 
入 记录 ， 完 成 发 布 操作 。 
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首页 | 招聘 信息 | 苍 U 祝 息 | 房 寻 全 外 | 求购 信息 | 六 吹 入 息 | 家 孝 芝 息 | 车 销 信 息 | 出 售 信息 | 拒 商 引 次 | 寻 人 / 寻 岁 自 示 


加 六 员 丽 | 及 条 页 | 联系 世 站”) 


易 查 供求 网 , 生活 更 轻松 


信息 类别 : -车辆 信息 ” 。 > * 请 正 确 先 择 全 要 发 布 的 信息 类 列 
信息 村 狠 : 阿 寺 得 

本 姐 颖 公司 有 一 批 好 车 出 稻 ， 各 种 车 型 应 有 尽 有 ,欢迎 三 
而 来 相册 < 


信息 为 育 : 
i 了 刁 系 人 : 茜 失 生 
关键 字 : | 联系 电 活 : [LU 
策 件 :求职 信息 - se 
则 联系 我 们 


易 坦 队 求 信息 网 we yel10_ con 税 仅 所 有 联系 电 渗 : 0431 045703** 


27.22 免费 供求 信息 发 布 网 页 


在 功能 导航 页 top.php 中 ， 在 “我 要 发 布 ”图 像 上 添加 图 像 映射 ， 并 绘制 热 区 。 其 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\InstanceW27\top.php) 


<map name="Map"> // 图 像 映射 
<area shape="rect" coords="356,8,424,28" href="release.php"> 
</map> 


在 信息 发 布 页 面 选 择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 为 了 避免 用 户 添加 空 信息 ， 在 
单 击 “ 发 布 信息 ”按钮 时 ， 应 用 JavaScript 脚本 自 定义 一 个 checkform0 函 数 ， 验 证 提交 的 表单 各 元 素 是 否 为 
空 值 ， 如 果 为 空 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\27\release_content.php) 

<script language="javascript"> 


function checkform(formX{ // 自 定义 一 个 JavaScript 函数 checkform() 
for(i=0;i<form.length:i++){ /应 用 for 循环 语句 检索 form 表单 元 素 的 值 是 否 为 空 
if(form.elements[i].value==™"){ // 如 果 form 表单 中 某 个 元 素 值 为 空 
alert(" 请 将 发 布 信息 填写 完整 ! "); // 弹 出 提示 信息 


@_ 


PHP+MySQL 开发 实战 


例如 ， 应 用 date0 函 数 格 式 化 一 个 日 期 ， 并 输出 日 期 的 值 。 其 代码 如 下 : 
<?php 

echo date("y:m:d"); 

?> 


结果 为 : 07:07:11。 
27.6.3 ”免费 供求 信息 发 布 模块 的 实现 过 程 


用 户 通过 单 击 前 台 页 面 导航 栏 的 “我 要 发 布 ” 超 链接 ， 进 入 信息 发 布 页 面 ， 如 图 27.22 所 示 。 程 序 会 先 


验证 用 户 输入 的 信息 ， 若 验证 失败 ， 则 返回 信息 发 布 页 面 ， 进 行 相应 提示 。 若 验证 成 功 ， 则 向 数据 库 中 插 
入 记录 ， 完 成 发 布 操作 。 


(个 C 易 查 供求 信息 网 CR 


首页 | 招聘 信息 | 苍 U 祝 息 | 房 寻 全 外 | 求购 信息 | 六 吹 入 息 | 家 孝 芝 息 | 车 销 信 息 | 出 售 信息 | 拒 商 引 次 | 寻 人 / 寻 岁 自 示 


加 六 员 丽 | 及 条 页 | 联系 世 站”) 


易 查 供求 网 , 生活 更 轻松 


信息 类别 : -车辆 信息 ” 。 > * 请 正 确 先 择 全 要 发 布 的 信息 类 列 
信息 村 狠 : 阿 寺 得 

本 姐 颖 公司 有 一 批 好 车 出 稻 ， 各 种 车 型 应 有 尽 有 ,欢迎 三 
而 来 相册 < 


信息 为 育 : 
i 了 刁 系 人 : 茜 失 生 
关键 字 : | 联系 电 活 : [LU 
策 件 :求职 信息 - se 
则 联系 我 们 


易 坦 队 求 信息 网 we yel10_ con 税 仅 所 有 联系 电 渗 : 0431 045703** 


27.22 免费 供求 信息 发 布 网 页 


在 功能 导航 页 top.php 中 ， 在 “我 要 发 布 ”图 像 上 添加 图 像 映射 ， 并 绘制 热 区 。 其 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\InstanceW27\top.php) 


<map name="Map"> // 图 像 映射 
<area shape="rect" coords="356,8,424,28" href="release.php"> 
</map> 


在 信息 发 布 页 面 选 择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 为 了 避免 用 户 添加 空 信息 ， 在 
单 击 “ 发 布 信息 ”按钮 时 ， 应 用 JavaScript 脚本 自 定义 一 个 checkform0 函 数 ， 验 证 提交 的 表单 各 元 素 是 否 为 
空 值 ， 如 果 为 空 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\27\release_content.php) 

<script language="javascript"> 


function checkform(formX{ // 自 定义 一 个 JavaScript 函数 checkform() 
for(i=0;i<form.length:i++){ /应 用 for 循环 语句 检索 form 表单 元 素 的 值 是 否 为 空 
if(form.elements[i].value==™"){ // 如 果 form 表单 中 某 个 元 素 值 为 空 
alert(" 请 将 发 布 信息 填写 完整 ! "); // 弹 出 提示 信息 


@_ 
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form.elements[].focus(); // 将 焦点 的 值 定位 到 为 空 值 的 表单 元 素 
return false; // 返 回 焦点 


} 


} 
</script> 
创建 与 数据 库 db_pursey 的 连接 ， 代 码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance27\conn\conn.php) 


<?php 

S$link=mysql_connect("localhost","root","111"); // 连 接 MySQL 服务 器 
mysql_select_db("db_pursey ",S$link); /连接 MySQL 数据 库 文件 db_pursey 
mysql_query("set names gb2312"); /采用 GB2312 编码 方式 

?> 


提交 表单 信息 到 数据 处 理 页 ， 应 用 insert…into 语句 向 免费 供求 信息 表 中 添加 供求 信息 。 如 果 信息 添加 
成 功 ， 则 弹出 成 功 的 信息 提示 ; 否则 弹出 失败 的 提示 信息 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\27\release_ok.php) 

<?php 


include("conn/conn.php"); // 连 接 数据 库 文件 

Stype=$_POST[type]; /获取 信息 类 型 

Stitle=$_POSTItitle]; // 获 取信 息 标题 

S$content=$_POST[content]; // 获 取信 息 内 容 

Slinkman=$_POSTIlinkman]; // 获 取 联 系 人 

S$tel=$_POSTI[tel]; // 获 取 联 系 电话 

© $edate=date("Y-m-d hii:s"); // 获 取 发 布 时 间 

© $sq=mysql_query("insert into tb_info(type,title,content,linkman,tel,checkstate,edate) 

values('$type','$title','$content','$linkman','$tel',0,'$edate')"); // 将 免费 的 供求 信息 添加 到 数据 表 中 

if($sql}{ // 如 果 添 加 操作 成 功 ， 则 弹出 提示 信息 
echo "<script>alert(' 恭 喜 您 ， 信 息 发 布 成 功 ! ');window.location.href='release.php';</script>"; 

Jelse{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert( 对 不 起 ， 信 息 发 布 失败 ! ');history.back();</script>"; 

上 

和 


Ni 
-说 明 。uate: 格式 化 一 个 本 地 时 间 / 日 期 。 
@ insert…into: 向 指定 的 数据 表 中 添加 数据 信息 。 


27.7 ”信息 检索 模块 设计 
多 中 视频 讲解 : 光盘 \TMIVVideo\ 第 27 章 \ 信 息 检索 模块 设计 .exe 


27.7.1 信息 检索 模块 概述 


信息 检索 是 对 已 存在 于 数据 库 中 的 数据 按 条 件 进行 筛选 浏览 ， 是 查看 历史 信息 和 确认 数据 操作 最 为 快 
速 、 有 效 的 办 法 。 信 息 检索 模块 主要 通过 选择 信息 类 型 和 输出 查询 关键 字模 糊 查询 供求 信息 资源 ， 并 输出 
查询 结果 。 考 虑 到 供求 信息 的 信息 量 较 大 ， 因 此 本 模块 对 与 查询 关键 字 相 匹配 的 查询 结果 进行 描 红 ， 从 而 
方便 用 户 的 浏览 。 信 息 检索 模块 的 示意 图 如 图 27.23 所 示 。 

@ 
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form.elements[].focus(); // 将 焦点 的 值 定位 到 为 空 值 的 表单 元 素 
return false; // 返 回 焦点 


} 


} 
</script> 
创建 与 数据 库 db_pursey 的 连接 ， 代 码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance27\conn\conn.php) 


<?php 

S$link=mysql_connect("localhost","root","111"); // 连 接 MySQL 服务 器 
mysql_select_db("db_pursey ",S$link); /连接 MySQL 数据 库 文件 db_pursey 
mysql_query("set names gb2312"); /采用 GB2312 编码 方式 

?> 


提交 表单 信息 到 数据 处 理 页 ， 应 用 insert…into 语句 向 免费 供求 信息 表 中 添加 供求 信息 。 如 果 信息 添加 
成 功 ， 则 弹出 成 功 的 信息 提示 ; 否则 弹出 失败 的 提示 信息 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\27\release_ok.php) 

<?php 


include("conn/conn.php"); // 连 接 数据 库 文件 

Stype=$_POST[type]; /获取 信息 类 型 

Stitle=$_POSTItitle]; // 获 取信 息 标题 

S$content=$_POST[content]; // 获 取信 息 内 容 

Slinkman=$_POSTIlinkman]; // 获 取 联 系 人 

S$tel=$_POSTI[tel]; // 获 取 联 系 电话 

© $edate=date("Y-m-d hii:s"); // 获 取 发 布 时 间 

© $sq=mysql_query("insert into tb_info(type,title,content,linkman,tel,checkstate,edate) 

values('$type','$title','$content','$linkman','$tel',0,'$edate')"); // 将 免费 的 供求 信息 添加 到 数据 表 中 

if($sql}{ // 如 果 添 加 操作 成 功 ， 则 弹出 提示 信息 
echo "<script>alert(' 恭 喜 您 ， 信 息 发 布 成 功 ! ');window.location.href='release.php';</script>"; 

Jelse{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 
echo "<script>alert( 对 不 起 ， 信 息 发 布 失败 ! ');history.back();</script>"; 

上 

和 


Ni 
-说 明 。uate: 格式 化 一 个 本 地 时 间 / 日 期 。 
@ insert…into: 向 指定 的 数据 表 中 添加 数据 信息 。 


27.7 ”信息 检索 模块 设计 
多 中 视频 讲解 : 光盘 \TMIVVideo\ 第 27 章 \ 信 息 检索 模块 设计 .exe 


27.7.1 信息 检索 模块 概述 


信息 检索 是 对 已 存在 于 数据 库 中 的 数据 按 条 件 进行 筛选 浏览 ， 是 查看 历史 信息 和 确认 数据 操作 最 为 快 
速 、 有 效 的 办 法 。 信 息 检索 模块 主要 通过 选择 信息 类 型 和 输出 查询 关键 字模 糊 查询 供求 信息 资源 ， 并 输出 
查询 结果 。 考 虑 到 供求 信息 的 信息 量 较 大 ， 因 此 本 模块 对 与 查询 关键 字 相 匹配 的 查询 结果 进行 描 红 ， 从 而 
方便 用 户 的 浏览 。 信 息 检索 模块 的 示意 图 如 图 27.23 所 示 。 

@ 
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设置 查询 条 件 


| 连接 数据 表 


根据 需要 组 合 检索 式 


27.23 ”信息 检索 模块 的 示意 图 


27.7.2 ”模糊 查询 技术 


在 对 数据 进行 查询 后 ， 最 终 需要 将 查询 结果 显示 在 页 面 中 反 
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在 利用 表格 显示 查询 结果 时 ， 通 常 是 将 查询 结果 保存 在 结果 
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面 给 出 其 实现 流程 图 ， 如 图 27.24 所 示 。 

考虑 到 用 户 不 可 能 全 面 了 解数 据 表 中 的 数据 信息 ， 如 不 能 确定 
所 要 查询 信息 的 内 容 、 查 询 的 主题 等 ， 这 时 就 需要 使 用 like 进行 模 
糊 查 询 。like 关键 字 需 要 使 用 通配符 在 字符 串 内 查找 指定 的 模式 ， 
所 以 读者 需要 了 解 通配符 及 其 含义 通配符 的 含义 如 表 27.5 所 示 。 


表 27.5 like 关键 字 中 的 通配符 及 说 明 


结果 集 是 
再 为 Null 
和 寅 | N 


+ 


输出 提示 信息 输出 一 条 数据 


图 27.24 信息 检索 模块 流程 图 


通 配 符 说 明 
% 由 零 个 或 更 多 字符 组 成 的 任意 字符 串 
任意 单个 字符 
[] | 用 于 指定 范围 ， 如 [A~F]， 表 示 A~ 下 内 的 任何 单个 字符 
加 表示 指定 范围 之 外 的 ， 如 [^ A~F] 表 示 A~F 以 外 的 任何 单个 字符 
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如 果 想 查询 包含 “女子 公寓 ”的 信息 ， 可 以 使 用 like 运算 符 配 合 通 配 符 “%” 完 成 。 其 SQL 语句 如 下 : 

select * from tb_info where content like '% 女 子 公 寅 %'; 

如 果 想 查找 信息 类 型 为 “公寓 信息 ”或 者 内 容 为 “女子 公寓 ”的 信息 时 配合 or 运算 符 来 使 用 。 其 SQL 
语句 如 下 : 

select * from tb_info ”where type=' 公 寓 信 息 ' or content like'% 女 子 公寓 %' 

本 模块 实现 付费 信息 与 查询 关键 字 相 匹配 的 信息 的 SQL 语句 如 下 

$type=$_POST[typel; /获取 信息 类 型 

$content=$_POSTIcontent]; 1/ 获取 查询 关键 字 

$sql1l=mysql_query("select * from tb_leaguerinfo where checkstate=1 and type='$type' and content 

like'%$content%' or title like'%$content%' or linkman like'%S$content%' or tel like'%$content%"); 

S$info1=mysql_fetch_array($sql1); /检索 付费 的 信息 

本 模块 实现 免费 信息 与 查询 关键 字 相 匹配 的 信息 的 SQL 语句 如 下 

$sql=mysql_query("select * from tb_info ”where checkstate=1 and type='$type' and content like'%$content%' or 

title like'%$content%' or linkman like'%$content%' or tel like'%$content%"); 

Sinfo=mysql_fetch_array($sql); /检索 免费 的 信息 


pa 


pe 


另外 ， 由 于 搜索 的 内 容 中 文字 比较 多 ， 为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 信 息 ， 所 以 在 搜索 引擎 
中 加 入 了 描 红 功 能 。 描 红 功 能 主要 用 str_ireplace0 函 数 实现 。 


27.7.3 “信息 检索 模块 的 实现 过 程 


在 开发 信息 检索 模块 时 ， 由 于 该 网 站 含有 大 量 的 数据 信息 ， 为 了 方便 用 户 浏览 网 站 信息 ， 需 要 添加 复 
合 条 件 查 询 实现 搜索 功能 。 在 信息 检索 区 的 “关键 字 ” 文 本 框 中 输入 和 欲 查询 的 关键 字 ， 在 “条 件 ” 下 拉 列 
表 框 中 选择 要 搜索 的 信息 类 型 ， 然 后 单 击 “ 开 始 搜索 ”按钮 ， 对 指定 条 件 的 记录 进行 检索 并 输出 结果 集 到 
浏览 器 ， 同 时 ， 为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 信息 ， 本 模块 对 查询 关键 字 进 行 了 描 红 。 运 行 结果 
如 图 27.25 所 示 。 


RC 遇 查 供求 信息 网 


Ha 共性 ， 芝 夫人. 天 用 全 这 多 有 ,和 二 
信和 ， 安全 后， 人 化 玫 浊 ,省 


| 贡 入 业 各 攻关 全 工作 二 这 


二 二 二 六 全 记得， 才 交 
a EE ,a 


27.25 ”信息 检索 页 面 的 运行 结果 
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pa 


pe 


另外 ， 由 于 搜索 的 内 容 中 文字 比较 多 ， 为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 信 息 ， 所 以 在 搜索 引擎 
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27.7.3 “信息 检索 模块 的 实现 过 程 
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合 条 件 查 询 实现 搜索 功能 。 在 信息 检索 区 的 “关键 字 ” 文 本 框 中 输入 和 欲 查询 的 关键 字 ， 在 “条 件 ” 下 拉 列 
表 框 中 选择 要 搜索 的 信息 类 型 ， 然 后 单 击 “ 开 始 搜索 ”按钮 ， 对 指定 条 件 的 记录 进行 检索 并 输出 结果 集 到 
浏览 器 ， 同 时 ， 为 了 方便 浏览 者 查找 自己 所 关注 的 内 容 信息 ， 本 模块 对 查询 关键 字 进 行 了 描 红 。 运 行 结果 
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Ha 共性 ， 芝 夫人. 天 用 全 这 多 有 ,和 二 
信和 ， 安全 后， 人 化 玫 浊 ,省 


| 贡 入 业 各 攻关 全 工作 二 这 


二 二 二 六 全 记得， 才 交 
a EE ,a 


27.25 ”信息 检索 页 面 的 运行 结果 


PHP+MySQL 开发 实战 


信息 检索 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 27.6 所 示 。 
表 27.6 ”信息 检索 页 面 所 涉及 的 重要 表单 元 素 


名 称 元 素 类 型 重要 属性 含义 
forml form method="post" action="findinfo.php" 表单 
content text id="content" size="20" 查询 关键 字 
<select name="type"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 -</option> 
type select <option value=" 求 职 信息 ”selected>- 求 职 信息 信息 类 型 
-</option> 
</select> 
search image src="Images/btn1 .gif" onClick="return chkinput(form)" 开始 搜索 按钮 


应 用 JavaScript 脚本 自 定义 一 个 chkinputO 函 数 ， 实 现 对 表单 提交 的 信息 进行 验证 。 其 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\27\left.php) 
<script language="javascript"> 


function chkinput(form)f // 自 定义 一 个 chkinput() 函 数 
if(form.content.value=="™" /| 判断 如 果 查 询 关键 字 文本 框 等 于 空 
alert(" 请 输入 查询 关键 字 !"); // 则 弹出 提示 信息 
form.content.select(); /重新 定位 焦点 
return false; // 返 回 表单 元 素 
中 
BE 
</script> 


将 表单 信息 提交 到 数据 处 理 页， 连接 数据 库 文件 ， 接 收 表 单 信息 ， 然 后 用 mysql_queryO 函 数 向 服务 器 
发 送 SQL 语句 ， 检 索 与 查询 关键 字 相 匹 配 的 信息 资源 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\27Vfindinfo.php) 


<?php 

include("conn/conn.php"); /| 连接 数据 库 文件 
$type=$_POSTItype]; // 获 取信 息 类 型 
$content=$_POSTI[content]; // 获 取 查 询 关 键 字 


$sql1=mysql_query("select * from tb_leaguerinfo where checkstate=1 and type='$type' and content 
like'%$content%' or title like'%$content%' or linkman like'%$content%' or tel like'%$content%"); 
$info1=mysql_fetch_array($sql1); /检索 付费 的 供求 信息 
$sql=mysql_query("select * from tb_info ”where checkstate=1 and type='$type' and content like'%$content%' or 
title like'%$content%' or linkman like'%$content%' or tel like'%$content%"); 

S$info=mysql_fetch_array($sql); // 检 索 免费 的 供求 信息 


Cp 


a 
日 
OC 培 明 信息 检索 需要 从 免费 信息 表 tb info 和 付费 信息 表 tb_leaguerinfo 中 获取 数据 ， 因 此 需要 向 
MySQL 服务 器 传递 两 条 SQL 语句 。 


do…while 循环 语句 输出 付费 信息 与 查询 关键 字 相 匹配 的 信息 资源 , 并 用 str_ireplace 函数 对 查询 关键 
字 实 现 描 红 功 能 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\27\findinfo.php) 
< 上 -下 面 输出 的 是 付费 信息 与 查询 关键 字 相 匹配 的 信息 -> 
<?php 
If($info1)f // 如 果 检 索 到 了 付费 信息 


@ 
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将 表单 信息 提交 到 数据 处 理 页， 连接 数据 库 文件 ， 接 收 表 单 信息 ， 然 后 用 mysql_queryO 函 数 向 服务 器 
发 送 SQL 语句 ， 检 索 与 查询 关键 字 相 匹 配 的 信息 资源 。 其 代码 如 下 : 
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include("conn/conn.php"); /| 连接 数据 库 文件 
$type=$_POSTItype]; // 获 取信 息 类 型 
$content=$_POSTI[content]; // 获 取 查 询 关 键 字 
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Cp 


a 
日 
OC 培 明 信息 检索 需要 从 免费 信息 表 tb info 和 付费 信息 表 tb_leaguerinfo 中 获取 数据 ， 因 此 需要 向 
MySQL 服务 器 传递 两 条 SQL 语句 。 


do…while 循环 语句 输出 付费 信息 与 查询 关键 字 相 匹配 的 信息 资源 , 并 用 str_ireplace 函数 对 查询 关键 
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If($info1)f // 如 果 检 索 到 了 付费 信息 


@ 


第 27 章 易 查 供求 信息 网 


dof // 则 用 do…while 循环 语句 输出 付费 信息 
?> 
<table width="540" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td height="26"> 
<!- 应 用 str_ireplace) 函 数 对 查询 关键 字 进 行 描 红 --> 
<!-- 对 与 查询 关键 字 所 匹配 的 信息 类 型 进行 描 红 --> 
[<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1 [type]);?>J 
nbsp; 

<!-- 对 与 查询 关键 字 所 匹配 的 信息 标题 进行 描 红 -> 
<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1 [title]);?>&nbsp; 


<!-- 对 与 查询 关键 字 所 匹配 的 发 布 时 间 进行 描 红 --> 

<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1 [edate]);?></td> 
</tr> 
<tr> 

<td height="26">&nbsp;&nbsp;&nbsp;&nbsp; 

<!-- 对 与 查询 关键 字 所 匹配 的 信息 内 容 进行 描 红 --> 

<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1 [content]);?> 


&nbsp; 


</td> 
</tr> 
<tr> 
<td height="26">&nbsp; 联 系 人 : 
<!-- 对 与 查询 关键 字 所 匹配 的 联系 人 进行 描 红 一 > 
<?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1 [linkman]);?> 


<!-- 对 与 查询 关键 字 所 匹配 的 联系 电话 进行 描 红 ”--> 
联系 电话 : <?php echo str_ireplace($content,"<font color=#FF0000'>".$content."</font>",$info1 


&nbsp; 


[tel]);?> 
< 
</td> 
</tr> 
</table> 
<?php 
}while($info1=mysql_fetch_array($sql1)); // 循 环 语句 结束 
?> 


oh 
en 代码 部 分 略 ， 详 见 本 书 附 赠 光 盘 。 


免费 信息 的 输出 方式 与 付费 信息 的 基本 类 似 ， 下 面 给 出 实现 过 程 的 核心 代码 结构 : 
(代码 位 置 ， 光 盘 \TM\Instance\27Vfindinfo.php) 


< 下 面 输 出 的 是 免费 信息 与 查询 关键 字 相 匹配 的 信息 -一 一 一 一 一 一 一 一 一 -> 
<?php 
if($infoX{ // 如 果 检 索 到 了 免费 信息 
do{ // 则 用 do…while 循环 语句 输出 付费 信息 
?> 
a /免费 信息 的 输出 方式 与 付费 信息 的 类 似 , 代码 略 
<?php 
}while($info=mysql_fetch_array($sql)); /do…while 循环 语句 结束 
} /所 条 件 语句 结束 
?> 


如 果 在 免费 信息 表 和 付费 信息 表 中 没有 检索 到 与 查询 关键 字 相 匹配 的 数据 ， 则 弹出 提示 信息 。 其 代码 


全 
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如 下 
(代码 位 置 ， 光盘 \TM\Instance\27\findinfo.php) 
<table width="540" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td align="center"> 您 检索 的 信息 资源 不 存在 ! </td> 


</tr> 
</table> 

27.8 后 台 首页 设计 
铬 il 视频 讲解 : 光盘 \TM\Video\ 第 27 章 \ 后 台 首页 设计 .exe 


27.8.1 ”后台 首页 概述 


程序 开发 人 员 在 设计 网 站 后 台 主页 时 ， 主 要 从 后 台 管 理 人 员 对 功能 的 易 操作 实用 性 、 网 站 的 易 维护 性 
专访 因由 来 用 村 柜 办 术 术 ，、 易 查 供求 信息 网 后 台 首 页 主要 包含 以 下 内 容 。 
发 布 付费 的 供求 信息 (包括 公寓 信息 、 招 聘 信息 、 pa 培训 信息 、 家 教 信息 、 房 屋 信息 、 
车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启示 等 ) ， 以 及 付费 信息 的 浏览 、 审 核 及 删 
除 功能 。 
回 ”免费 信息 的 浏览 、 审 核 及 删除 功能 。 
回 ”企业 广告 信息 的 发 布 、 浏 览 、 前 台 推荐 显示 、 删 除 功能 。 
回 ”网 站 首页 : 为 管理 员 进 入 前 台 提供 一 个 入 口 。 

回 ”退出 登录 : 注销 当前 用 户 。 
下 面 看 一 下 本 案例 中 提供 的 后 台 首 页 , 该 页 面 在 本 书 光盘 中 的 路 径 为 TM\27\admin\index.php, 如 图 27.26 
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图 27.26 ”供求 信息 网 站 后 台 首 页 
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如 下 
(代码 位 置 ， 光盘 \TM\Instance\27\findinfo.php) 
<table width="540" border="0" cellspacing="0" cellpadding="0"> 
<tr> 
<td align="center"> 您 检索 的 信息 资源 不 存在 ! </td> 


</tr> 
</table> 

27.8 后 台 首页 设计 
铬 il 视频 讲解 : 光盘 \TM\Video\ 第 27 章 \ 后 台 首页 设计 .exe 


27.8.1 ”后台 首页 概述 


程序 开发 人 员 在 设计 网 站 后 台 主页 时 ， 主 要 从 后 台 管 理 人 员 对 功能 的 易 操作 实用 性 、 网 站 的 易 维护 性 
专访 因由 来 用 村 柜 办 术 术 ，、 易 查 供求 信息 网 后 台 首 页 主要 包含 以 下 内 容 。 
发 布 付费 的 供求 信息 (包括 公寓 信息 、 招 聘 信息 、 pa 培训 信息 、 家 教 信息 、 房 屋 信息 、 
车 辆 信息 、 求 购 信息 、 出 售 信息 、 招 商 引资 、 寻 人 / 物 启示 等 ) ， 以 及 付费 信息 的 浏览 、 审 核 及 删 
除 功能 。 
回 ”免费 信息 的 浏览 、 审 核 及 删除 功能 。 
回 ”企业 广告 信息 的 发 布 、 浏 览 、 前 台 推荐 显示 、 删 除 功能 。 
回 ”网 站 首页 : 为 管理 员 进 入 前 台 提供 一 个 入 口 。 

回 ”退出 登录 : 注销 当前 用 户 。 
下 面 看 一 下 本 案例 中 提供 的 后 台 首 页 , 该 页 面 在 本 书 光盘 中 的 路 径 为 TM\27\admin\index.php, 如 图 27.26 


轨 ， 


PETTTTEEE 
友 而 从 


E] 挛 信息 内 春 bm 
eave 长 了 六 步行 过 并 业 ， 环 
a 
ET rp 澳 ， 电 洒 笨 ， 仙 水 信 ， 夫 胡 oe- jt /wpe 
faa: -2 EE] | | mx 
本 付费 信息 显示 方式 本 再 ， dn 
旦 ,和 光束 浊 入 子 入 位 
安 付 风 杖 而 a 过， 挤 人 2 人 局, 8 人 
个 BH 个 不 人 几 “全部 同 * 交 朝 名利, 室内 地, 要 者 到 内 
bb 自流 妇 于 公 竟 。。 好, 用 于 基 ， 有 志 祝 ,洪水 - 从 王 网 绪 。 ebeevee zn0T-i2-20 09-n-53 已 布 模 而 /WW 
属于 .条 全 化 专人 过 香 ， 是 兴 叶 站 公 
信息 叫 : “到 重 信息 ”| 下 by 


[SS 
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图 27.26 ”供求 信息 网 站 后 台 首 页 
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27.8.2 frame 框架 技术 


易 查 供求 信息 网 后 台 采 用 框架 技术 进行 页 面 布 局 。 所 谓 框架 就 是 网 页 的 各 部 分 为 相互 独立 的 网 页 ， 又 
由 一 个 网 页 将 这 些 分 开 的 网 页 组 成 一 个 完整 的 网 页 ， 显 示 在 浏览 者 的 浏览 器 中 ， 重 复出 现 的 内 容 被 固定 下 
来 ， 每 次 浏览 者 发 出 对 页 面 的 请 求 时 ， 只 下 载 发 生变 化 的 框架 页 面 ， 其 他 子 页 面 保持 不 变 。 

使 用 框架 可 以 将 容器 窗口 划分 为 若干 个 子 窗口 ， 在 每 个 子 窗口 可 以 分 别 显 示 不 同 的 网 页 。 首 先 在 
Dreamweaver 8 中 创建 一 个 “ 左 一 中 一 右 ”的 框架 集 ， 然 后 在 标识 中 添加 “上 一 中 一 下 ”的 框架 集 ， 最 后 
在 标识 @ 中 添加 一 个 “ 左 一 右 ” 的 框架 集 ， 从 而 完成 一 个 完整 的 后 台 框 架 。 构建 框架 的 流程 如 图 27.27 所 示 。 


27.27 网 站 后 台 框 架 流程 
使 用 框架 可 以 非常 方便 地 完成 导航 工作 。 下 面 详细 介绍 框架 的 基本 结构 、 设 置 框架 集 属 性 和 框架 属性 。 
1. 框架 网 页 的 基本 结构 
框架 网 页 通过 一 个 或 多 个 FRAMESET 和 FRAME 标记 来 定义 。 在 框架 网 页 中 ， 将 FRAMESET 标记 置 
于 HEAD 标记 后 , 以 取代 BODY 标记 的 位 置 , 当 客 户 端 浏览 器 不 支持 框架 网 页 时 , 还 可 以 使 用 NOFRAMES 
标记 给 出 框架 不 能 被 显示 时 的 蔡 换 内 容 。 框 架 网 页 的 基本 结构 可 以 表示 如 下 : 
<html> 
<head> 
<title> 基 本 框架 页 </title> 
</head> 
<frameset> 
<frame> 
<frame> 
</frameset> 
<noframes> 
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<body> 
对 不 起 ! 您 的 浏览 器 不 支持 框架 页 面 的 显示 ! 
</body> 
</noframes> 
</html> 
2. 设置 框架 集 的 属性 
框架 集 包含 如 何 组 织 各 个 框架 的 信息 , 可 以 通过 FRAMESET 标记 来 定义 ,框架 是 按照 行 和 列 来 组 织 的 ， 
可 以 使 用 FRAMESET 标记 的 下 列 属性 对 框架 的 结构 进行 设置 。 
(1) 左右 分 割 窗口 属性 COLS 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 左右 分 割 窗口 属性 COLS 实现 。 其 语法 格式 
如 下 : 
<frameset cols="Value,value,..."> 
<frame> 
<frame> 
</frameset> 
参数 value 用 于 指定 各 个 框架 的 列 宽 ， 取 值 有 像素 、 百 分 比 〈%) 和 相对 尺寸 (*) 3 种 形式 。 
例如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 列 ， 其 中 第 1 列 占 浏览 器 窗口 宽度 的 20%， 第 2 列 为 120 
像素 ， 第 3 列 为 浏览 器 窗口 的 剩余 部 分 。 其 代码 如 下 : 
<frameset cols="20%,120,*" > 
<frame> 
<frame> 
</frameset> 


Ee 


技巧 如 果 将 COLS 属性 设置 为 “*，*，*”， 则 表示 将 窗口 划分 成 3 个 等 宽 的 框架 ; 如 果 将 COLS 
属性 设置 为 “*，2*，3*”， 则 表示 左边 的 框架 占 窗口 宽度 的 116， 中 间 的 框架 占 窗口 宽度 的 1/3， 右 边 
的 框架 占 窗口 宽度 的 1/2。 


(2) 上 下 分 割 窗 口 属性 ROWS 
在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 上 下 分 割 窗口 属性 ROWS 实现 。 其 语法 格式 
如 下 : 
<frameset rows="Value,value,..."> 
<frame> 
<frame> 
</frameset> 
value 用 于 指定 各 个 框架 的 行 高 , 取 值 有 像素 、 百分比 (%) 和 相对 尺寸 (*) 3 种 形式 , 设置 方法 与 COLS 
属性 类 似 。 
例如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 行 ， 其 中 的 第 1 行 占 浏览 器 窗口 宽度 的 20%, 第 2 行为 120 
像素 ， 第 3 行为 浏览 器 窗口 的 剩余 部 分 的 框架 。 其 代码 如 下 : 
<frameset rows="20%,120,*" > 
<frame> 
<frame> 
</frameset> 
(3) 框架 边框 显示 属性 FRAMEBORDER 
该 属性 用 于 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框 ， 默 认 值 ) 或 0 (不 显示 边框 ) 。 
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<body> 
对 不 起 ! 您 的 浏览 器 不 支持 框架 页 面 的 显示 ! 
</body> 
</noframes> 
</html> 
2. 设置 框架 集 的 属性 
框架 集 包含 如 何 组 织 各 个 框架 的 信息 , 可 以 通过 FRAMESET 标记 来 定义 ,框架 是 按照 行 和 列 来 组 织 的 ， 
可 以 使 用 FRAMESET 标记 的 下 列 属性 对 框架 的 结构 进行 设置 。 
(1) 左右 分 割 窗口 属性 COLS 
在 水 平方 向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 左右 分 割 窗口 属性 COLS 实现 。 其 语法 格式 
如 下 : 
<frameset cols="Value,value,..."> 
<frame> 
<frame> 
</frameset> 
参数 value 用 于 指定 各 个 框架 的 列 宽 ， 取 值 有 像素 、 百 分 比 〈%) 和 相对 尺寸 (*) 3 种 形式 。 
例如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 列 ， 其 中 第 1 列 占 浏览 器 窗口 宽度 的 20%， 第 2 列 为 120 
像素 ， 第 3 列 为 浏览 器 窗口 的 剩余 部 分 。 其 代码 如 下 : 
<frameset cols="20%,120,*" > 
<frame> 
<frame> 
</frameset> 


Ee 


技巧 如 果 将 COLS 属性 设置 为 “*，*，*”， 则 表示 将 窗口 划分 成 3 个 等 宽 的 框架 ; 如 果 将 COLS 
属性 设置 为 “*，2*，3*”， 则 表示 左边 的 框架 占 窗口 宽度 的 116， 中 间 的 框架 占 窗口 宽度 的 1/3， 右 边 
的 框架 占 窗口 宽度 的 1/2。 


(2) 上 下 分 割 窗 口 属性 ROWS 
在 垂直 方向 上 将 浏览 器 分 割 成 多 个 窗口 ， 可 以 通过 框架 的 上 下 分 割 窗口 属性 ROWS 实现 。 其 语法 格式 
如 下 : 
<frameset rows="Value,value,..."> 
<frame> 
<frame> 
</frameset> 
value 用 于 指定 各 个 框架 的 行 高 , 取 值 有 像素 、 百分比 (%) 和 相对 尺寸 (*) 3 种 形式 , 设置 方法 与 COLS 
属性 类 似 。 
例如 ， 若 要 通过 框架 将 浏览 器 窗口 划分 为 3 行 ， 其 中 的 第 1 行 占 浏览 器 窗口 宽度 的 20%, 第 2 行为 120 
像素 ， 第 3 行为 浏览 器 窗口 的 剩余 部 分 的 框架 。 其 代码 如 下 : 
<frameset rows="20%,120,*" > 
<frame> 
<frame> 
</frameset> 
(3) 框架 边框 显示 属性 FRAMEBORDER 
该 属性 用 于 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 边框 ， 默 认 值 ) 或 0 (不 显示 边框 ) 。 
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(4) FRAMESPACING 
该 属性 用 于 指定 框架 之 间 的 间隔 ， 以 像素 为 单位 。 如 果 不 设置 该 属性 ， 则 框架 之 间 没 有 间隔 。 
(5) 指定 边框 宽度 属性 BORDER 

该 属性 用 于 指定 边框 的 宽度 ， 只 有 FRAMEBORDER 属性 为 1 时 有 效 。 


3. 设置 框架 的 属性 

使 用 FRAME 标记 可 以 设置 框架 的 属性 , 包括 框架 的 名 称 、 框架 是 否 包 含 深 动 条 以 及 在 框架 中 显示 的 网 
页 等 。 其 语法 格式 如 下 : 

<frame name=" 框 架 名 称 " src=" 文 件 " frameborder=" 数 值 " scrolling=" 值 " [noresize] > 
属性 说 明 如 下 。 
name: 指定 框架 的 名 称 。 
src: 指定 在 框架 中 显示 的 网 页 文件 (包括 HTML、ASP 等 网 页 文件 ) 。 
frameborder: 指定 框架 周围 是 否 显示 边框 ， 取 值 为 1 (显示 ) 或 0 (不 显示 ) 。 默 认 值 为 1。 
scrolling: 指定 框架 是 否 包含 滚动 条 。 如 果 将 该 属性 设置 为 yes， 则 框架 包含 滚动 条 ; 若 将 该 属性 
设置 为 no， 则 框架 不 包含 滚动 条 ， 如 果 将 该 属性 设置 为 aato， 则 在 需要 时 包含 滚动 条 。 
noresize: 可 选 属性 ， 若 指定 了 该 属性 ， 则 不 能 调整 框架 的 大 小 。 


办 办 办 


加 


27.8.3 后台 首页 的 实现 过 程 


根据 以 上 两 节 的 页 面 概述 及 实现 技术 分 析 ， 需 要 分 别 创建 实现 各 区 域 的 PHP 文件 ， 如 实现 Banner 广告 
栏 的 top.php、 功 能 导航 栏 的 left.php、 内 容 显示 区 的 main.php 和 页 尾 文 件 bottom.php 等 。 下 面 给 出 实现 该 系 
统 后 台 框 架 布局 的 完整 代码 。 
(代码 位 置 ， 光 盘 \TM\Instance\27\admin\index.php) 
<frameset rows="*" cols="1*,16005,1*" framespacing="0" frameborder="NO" border="0"> 
<frame src="blank.php" name="left" scrolling="NO" noresize> <!-- 设 置 空 框架 页 --> 
<frameset rows="1005,” cols="*" framespacing="0" frameborder="NO" border="0"> 
<frameset rows="94,*,190" cols="*" framespacing="0" frameborder="NO" border="0"> 
<frame src="top.php" name="topFrame" scrolling="NO" noresize> 
<frameset rows="*" cols="229,*" framespacing="0" frameborder="NO" border="0"> 
<frame src="left.php" name="leftFrame'" scrolling="NO" noresize> 
<frame src="main.php" name="mainFrame" scrolling="NO" noresize> 
</frameset> 
<frame src="bottom.php" name="bottomFrame" scrolling="NO" noresize> 
</frameset> 
<frame src="blank.php" name="right" scrolling="NO" noresize> 
</frameset> 

<frame src="blank.php"></frameset> <!-- 设 置 空 框架 页 --> 
<noframes><body> 
</body></noframes> 


站 在 建设 Web 网 站 时 ， 如 何 让 不 同 分 辩 率 的 用 户 都 能 看 到 网 页 的 最 佳 效 果 是 程序 员 在 设计 之 初 
所 要 考虑 的 首要 问题 。 为 了 使 屏蔽 的 分 辨 率 在 大 于 1024 x 768 像素 的 设置 时 仍然 处 于 居中 显示 ， 只 需要 
在 设置 框架 布局 时 ， 在 主 框架 两 侧 各 设置 一 个 宽度 相同 的 blank php 空 页 即 可 。 

-加 
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27.9 ”付费 供求 信息 发 布 模块 设计 
峰 4 视频 讲解 : 光盘 \TM\Video\ 第 27 章 \ 付 费 供求 信息 发 布 模块 设计 .exe 


27.9.1 付费 供求 信息 发 布 模块 概述 


付费 供求 信息 发 布 是 供求 信息 网 站 非常 重要 的 功能 ， 也 是 供求 信 
息 网 站 的 盈利 点 。 企 业 或 用 户 可 以 根据 自身 需要 对 供求 信息 先进 行 付 
费 ， 付 费 后 由 管理 员 在 后 台 将 供求 信息 发 布 到 相应 的 信息 类 别 中 〈 共 
包括 11 个 信息 类 别 : 招聘 信息 、 求 职 信息 、 培 训 信 息 、 公 寓 信 息 、 家 
教 信息 、 车 辆 信息 、 物 品 求购 、 物 品 出 售 、 求 兑 出 况 、 寻 求 合 作 、 企 
业 广 告 ) 。 供 求 信息 成 功 发 布 后 ， 管 理 员 需 要 在 后 台 对 发 布 的 供求 信 
息 进行 审核 , 如 果 审核 通过 后 , 则 显示 在 前 台 相 应 的 信息 类 别 网 页 中 。 
付费 供求 信息 发 布 的 流程 如 图 27.28 所 示 。 


27.9.2 ”计算 供求 信息 的 有 效 时 间 


付费 供求 信息 与 免费 供求 信息 不 同 的 是 ， 付 费 供求 信息 不 仅 需要 
收取 一 定 的 费用 ， 而 且 还 需要 一 定 的 时 间 限 制 ， 例 如 ， 网 站 要 求 一 个 
月 〈 按 30 天 计算 ) 每 条 信息 交 10 元 的 信息 费 ， 如 果 用 户 交纳 20 元 ， 
那么 信息 显示 的 天 数 就 是 60 天 。 在 前 台 进行 显示 时 , 不 需要 管理 员 进 
行 手动 管理 ， 而 是 通过 程序 直接 计算 出 信息 显示 的 截止 时 间 。 

信息 显示 的 截止 时 间 = 系统 当前 日 期 + 信息 的 有 效 天 数 〈 与 用 户 交 


纳 的 信息 费 相关 ) 。 图 27.28 ”付费 供求 信息 发 布 流程 图 
自动 计算 信息 显示 的 截止 时 间 的 具体 代码 如 下 : 
$days=$_POST[days]; // 通 过 表单 传 值 获取 信息 显示 的 天 数 
$showday=date("Y-m-d",(time()+3600*24*$days)); // 信 息 显示 的 截止 时 间 


A 
A 涪 明 信息 的 有 效 天 数 与 用 户 交纳 的 信息 费 相 关 ， 交 党 不 通过 本 程序 完成 ， 因 此 ， 信 息 的 有 效 天 数 需 
要 管理 员 手 动 添加 。 


27.9.3 ”付费 供求 信息 发 布 模块 的 实现 过 程 


户 通过 单 击 页 面 导 航 区 的 “付费 信息 ” 超 链接 ， 进 入 付费 信息 发 布 页 面 ， 如 图 27.29 所 示 。 填 写真 实 
有 效 的 付费 信息 ， 单 击 “发 布 信息 ”按钮 ， 程 序 会 先 验证 用 户 输入 的 信息 ， 若 验证 失败 ， 则 返回 信息 发 布 
页 面 ， 进 行 相应 提示 。 若 验证 成 功 ， 则 向 数据 库 中 插入 记录 ， 完 成 付费 信息 的 发 布 操作 。 

在 左 侧 框架 leftphp 页 中 ， 添 加 “付费 信息 ”图 像 域 及 表单 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance27\admin\left.php) 

<form name="form1" method="post" action="release_content.php" target="mainFrame"> 
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收取 一 定 的 费用 ， 而 且 还 需要 一 定 的 时 间 限 制 ， 例 如 ， 网 站 要 求 一 个 
月 〈 按 30 天 计算 ) 每 条 信息 交 10 元 的 信息 费 ， 如 果 用 户 交纳 20 元 ， 
那么 信息 显示 的 天 数 就 是 60 天 。 在 前 台 进行 显示 时 , 不 需要 管理 员 进 
行 手动 管理 ， 而 是 通过 程序 直接 计算 出 信息 显示 的 截止 时 间 。 
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自动 计算 信息 显示 的 截止 时 间 的 具体 代码 如 下 : 
$days=$_POST[days]; // 通 过 表单 传 值 获取 信息 显示 的 天 数 
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有 效 的 付费 信息 ， 单 击 “发 布 信息 ”按钮 ， 程 序 会 先 验证 用 户 输入 的 信息 ， 若 验证 失败 ， 则 返回 信息 发 布 
页 面 ， 进 行 相应 提示 。 若 验证 成 功 ， 则 向 数据 库 中 插入 记录 ， 完 成 付费 信息 的 发 布 操作 。 

在 左 侧 框架 leftphp 页 中 ， 添 加 “付费 信息 ”图 像 域 及 表单 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance27\admin\left.php) 
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收取 一 定 的 费用 ， 而 且 还 需要 一 定 的 时 间 限 制 ， 例 如 ， 网 站 要 求 一 个 
月 〈 按 30 天 计算 ) 每 条 信息 交 10 元 的 信息 费 ， 如 果 用 户 交纳 20 元 ， 
那么 信息 显示 的 天 数 就 是 60 天 。 在 前 台 进行 显示 时 , 不 需要 管理 员 进 
行 手动 管理 ， 而 是 通过 程序 直接 计算 出 信息 显示 的 截止 时 间 。 

信息 显示 的 截止 时 间 = 系统 当前 日 期 + 信息 的 有 效 天 数 〈 与 用 户 交 


纳 的 信息 费 相关 ) 。 图 27.28 ”付费 供求 信息 发 布 流程 图 
自动 计算 信息 显示 的 截止 时 间 的 具体 代码 如 下 : 
$days=$_POST[days]; // 通 过 表单 传 值 获取 信息 显示 的 天 数 
$showday=date("Y-m-d",(time()+3600*24*$days)); // 信 息 显示 的 截止 时 间 
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A 涪 明 信息 的 有 效 天 数 与 用 户 交纳 的 信息 费 相 关 ， 交 党 不 通过 本 程序 完成 ， 因 此 ， 信 息 的 有 效 天 数 需 
要 管理 员 手 动 添加 。 


27.9.3 ”付费 供求 信息 发 布 模块 的 实现 过 程 


户 通过 单 击 页 面 导 航 区 的 “付费 信息 ” 超 链接 ， 进 入 付费 信息 发 布 页 面 ， 如 图 27.29 所 示 。 填 写真 实 
有 效 的 付费 信息 ， 单 击 “发 布 信息 ”按钮 ， 程 序 会 先 验证 用 户 输入 的 信息 ， 若 验证 失败 ， 则 返回 信息 发 布 
页 面 ， 进 行 相应 提示 。 若 验证 成 功 ， 则 向 数据 库 中 插入 记录 ， 完 成 付费 信息 的 发 布 操作 。 

在 左 侧 框架 leftphp 页 中 ， 添 加 “付费 信息 ”图 像 域 及 表单 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance27\admin\left.php) 
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付费 供求 信息 与 免费 供求 信息 不 同 的 是 ， 付 费 供求 信息 不 仅 需要 
收取 一 定 的 费用 ， 而 且 还 需要 一 定 的 时 间 限 制 ， 例 如 ， 网 站 要 求 一 个 
月 〈 按 30 天 计算 ) 每 条 信息 交 10 元 的 信息 费 ， 如 果 用 户 交纳 20 元 ， 
那么 信息 显示 的 天 数 就 是 60 天 。 在 前 台 进行 显示 时 , 不 需要 管理 员 进 
行 手动 管理 ， 而 是 通过 程序 直接 计算 出 信息 显示 的 截止 时 间 。 

信息 显示 的 截止 时 间 = 系统 当前 日 期 + 信息 的 有 效 天 数 〈 与 用 户 交 


纳 的 信息 费 相关 ) 。 图 27.28 ”付费 供求 信息 发 布 流程 图 
自动 计算 信息 显示 的 截止 时 间 的 具体 代码 如 下 : 
$days=$_POST[days]; // 通 过 表单 传 值 获取 信息 显示 的 天 数 
$showday=date("Y-m-d",(time()+3600*24*$days)); // 信 息 显示 的 截止 时 间 
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A 涪 明 信息 的 有 效 天 数 与 用 户 交纳 的 信息 费 相 关 ， 交 党 不 通过 本 程序 完成 ， 因 此 ， 信 息 的 有 效 天 数 需 
要 管理 员 手 动 添加 。 


27.9.3 ”付费 供求 信息 发 布 模块 的 实现 过 程 


户 通过 单 击 页 面 导 航 区 的 “付费 信息 ” 超 链接 ， 进 入 付费 信息 发 布 页 面 ， 如 图 27.29 所 示 。 填 写真 实 
有 效 的 付费 信息 ， 单 击 “发 布 信息 ”按钮 ， 程 序 会 先 验证 用 户 输入 的 信息 ， 若 验证 失败 ， 则 返回 信息 发 布 
页 面 ， 进 行 相应 提示 。 若 验证 成 功 ， 则 向 数据 库 中 插入 记录 ， 完 成 付费 信息 的 发 布 操作 。 

在 左 侧 框架 leftphp 页 中 ， 添 加 “付费 信息 ”图 像 域 及 表单 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance27\admin\left.php) 

<form name="form1" method="post" action="release_content.php" target="mainFrame"> 
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<input name="imageField" type="image" class="input1" src="images/btn_fufei.gif" width="210" height="39" 
border="0"> 
</form> 


PE Bre 


图 27.29 付费 供求 信息 发 布 页 面 运行 结果 


单 击 “ 付 费 信息 ”按钮 ， 将 信息 页 release_content.php 中 的 内 容 显示 在 框架 显示 页 mainFrame 中 。 
付费 供求 信息 发 布 页 面 主要 用 于 发 布 付费 的 供求 信息 , 该 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 27.7 所 示 。 


表 27.7 ”付费 供求 信息 页 面 所 涉及 的 重要 表单 元 素 


="post" action="release_ok.php 
<select name="type"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 -</option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 -</option> 


信息 类 型 


<option value=" 寻 人 / 物 启示 ">- 寻 人 / 物 启示 -</option> 
</select> 


class="inputl" Value="1" checked 
EE EE 


Cols="55" rows="8" 


“是 否 付费 ” 复 选 框 


g" onClick="retum checkformform): 


在 付费 信息 发 布 页 面 选择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 ah 
在 单 击 “ 发 布 信息 ”按钮 时 ， 应 用 JavaScript 脚本 自 定义 一 个 checkform0 函 数 ， 验 证 提交 的 表单 各 元 素 是 


为 空 值 ， 如 果 为 空 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。 提 交 pp ie 
应 用 insert…into 语句 向 免费 供求 信息 表 中 添加 供求 信息 。 如 果 信 息 添加 成 功 ， 则 弹出 成 功 的 信息 提示 ; 否 
则 弹出 失败 的 提示 信息 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance27\admin\release_ok.php) 

<?php 
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图 27.29 付费 供求 信息 发 布 页 面 运行 结果 


单 击 “ 付 费 信息 ”按钮 ， 将 信息 页 release_content.php 中 的 内 容 显示 在 框架 显示 页 mainFrame 中 。 
付费 供求 信息 发 布 页 面 主要 用 于 发 布 付费 的 供求 信息 , 该 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 27.7 所 示 。 
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<option value=" 招 聘 信息 ">- 招 聘 信息 -</option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 -</option> 


信息 类 型 


<option value=" 寻 人 / 物 启示 ">- 寻 人 / 物 启示 -</option> 
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g" onClick="retum checkformform): 


在 付费 信息 发 布 页 面 选择 要 发 布 的 信息 类 型 后 ， 填 写真 实 有 效 的 供求 信息 。 ah 
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为 空 值 ， 如 果 为 空 ， 则 弹出 提示 信息 ， 并 将 焦点 定位 到 为 空 值 的 表单 元 素 。 提 交 pp ie 
应 用 insert…into 语句 向 免费 供求 信息 表 中 添加 供求 信息 。 如 果 信 息 添加 成 功 ， 则 弹出 成 功 的 信息 提示 ; 否 
则 弹出 失败 的 提示 信息 。 其 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance27\admin\release_ok.php) 

<?php 


图 
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Include("../conn/conn.php"); 1/ 连 接 数 据 库 文 件 
S$type=$_POST[type]; // 获 取信 息 类 型 
$flag=$_POSTIflag]:; /获取 付款 状态 
Stitle=$_POST[title]: /获取 信息 标题 
$content=$_POSTI[content]; /获取 信息 内 容 
S$linkman=$_POSTIlinkman]; /获取 联系 人 
$days=$_POST[days]; // 获 取 发 布 时 间 
Stel=$_POSTItel]: /获取 联 系 电话 
S$sdate=date("Y-m-d"); /当前 系统 时 间 

@ $showday=date("Y-m-d",(time()+3600*24*$days)); // 获 取信 息 的 有 效 时 间 


$sql=mysql_query("insert into tb_leaguerinfotype,,title,content,linkman ,tel,sdate,showday,checkstate) 
values('$type','$title','$content','$linkman','$tel','$sdate','$showday', $flag)"); // 将 付费 的 供求 信息 添加 到 数据 表 中 


if($sql}{ // 如 果 添 加 操作 成 功 ， 则 弹出 提示 信息 
@ ”echo "<script>alert(' 信 息 发 布 成 功 ! '); parent.mainFrame.location.href='release_content.php';</script>"; 
}else{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 


echo "<script>alert(' 信 息 发 布 失败 ! ');history.back();</script>"; 
} 


2 


a 
和 培 明 @ date("Y-m-d",(time(+3600*24*$days)): 信息 的 有 效 时 间 = 当 前 期 日 期 + 付费 期 限 . 应 用 time0 
函数 获取 当前 日 期 时 间 蕉 ,付费 期 限 的 时 间 戳 等 于 3600 秒 x24 小 时 x 指定 天 数 ， 并 通过 date0 函 数 格式 
化 为 指定 日 期 格式 。 
@ parentmainFrame location .href-'release_contentphp': 刷新 父 框架 页 release_content.php 中 的 信息 。 


27.10 付费 信息 管理 模块 设计 
名 0 视频 讲解 :光盘 \TM\Video\ 第 27 章 \ 付 费 信息 管理 模块 设计 .exe 


27.10.1 付费 信息 管理 模块 概述 


付费 信息 管理 模块 主要 包括 查看 图 书 列表 、 付 费 信息 审核 和 付费 信息 删除 3 个 功能 。 付 费 信息 管理 模 
块 的 框架 如 图 27.30 所 示 。 


付费 信息 管理 


[Es | 


+ 


付费 信息 列表 


+ 


| [ass 


小 
油 
许 
导 
只 
和 
ll 


27.30 ”付费 信息 管理 模块 的 框架 图 
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Include("../conn/conn.php"); 1/ 连 接 数 据 库 文 件 
S$type=$_POST[type]; // 获 取信 息 类 型 
$flag=$_POSTIflag]:; /获取 付款 状态 
Stitle=$_POST[title]: /获取 信息 标题 
$content=$_POSTI[content]; /获取 信息 内 容 
S$linkman=$_POSTIlinkman]; /获取 联系 人 
$days=$_POST[days]; // 获 取 发 布 时 间 
Stel=$_POSTItel]: /获取 联 系 电话 
S$sdate=date("Y-m-d"); /当前 系统 时 间 

@ $showday=date("Y-m-d",(time()+3600*24*$days)); // 获 取信 息 的 有 效 时 间 


$sql=mysql_query("insert into tb_leaguerinfotype,,title,content,linkman ,tel,sdate,showday,checkstate) 
values('$type','$title','$content','$linkman','$tel','$sdate','$showday', $flag)"); // 将 付费 的 供求 信息 添加 到 数据 表 中 


if($sql}{ // 如 果 添 加 操作 成 功 ， 则 弹出 提示 信息 
@ ”echo "<script>alert(' 信 息 发 布 成 功 ! '); parent.mainFrame.location.href='release_content.php';</script>"; 
}else{ // 如 果 添 加 操作 失败 ， 则 弹出 提示 信息 


echo "<script>alert(' 信 息 发 布 失败 ! ');history.back();</script>"; 
} 


2 


a 
和 培 明 @ date("Y-m-d",(time(+3600*24*$days)): 信息 的 有 效 时 间 = 当 前 期 日 期 + 付费 期 限 . 应 用 time0 
函数 获取 当前 日 期 时 间 蕉 ,付费 期 限 的 时 间 戳 等 于 3600 秒 x24 小 时 x 指定 天 数 ， 并 通过 date0 函 数 格式 
化 为 指定 日 期 格式 。 
@ parentmainFrame location .href-'release_contentphp': 刷新 父 框架 页 release_content.php 中 的 信息 。 


27.10 付费 信息 管理 模块 设计 
名 0 视频 讲解 :光盘 \TM\Video\ 第 27 章 \ 付 费 信息 管理 模块 设计 .exe 


27.10.1 付费 信息 管理 模块 概述 


付费 信息 管理 模块 主要 包括 查看 图 书 列表 、 付 费 信息 审核 和 付费 信息 删除 3 个 功能 。 付 费 信息 管理 模 
块 的 框架 如 图 27.30 所 示 。 
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27.30 ”付费 信息 管理 模块 的 框架 图 
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27.10.2 ”数据 的 更 新 和 删除 技术 


付费 信息 管理 页 面 在 实现 信息 审核 及 删除 的 功能 时 应 用 到 了 UPDATE 更 新 语句 和 DELETE 删除 语句 。 
下 面 对 这 两 个 语句 进行 详细 地 讲解 。 
1. UPDATE 语句 
UPDATE 语句 用 来 改变 单行 上 的 一 列 或 多 列 的 值 ， 或 者 改变 单个 表 中 选 定 的 一 些 行 上 的 多 个 列 值 。 
UPDATE 语句 的 语法 如 下 : 
UPDATE<table_name | view_name> 
SET <column_name>=<expression> 
[.…,<last column_name>=<last expression>] 
[WHERE<search_condition>] 
UPDATE 语句 的 参数 说 明 如 表 27.8 所 示 。 


参数 


table_ name 


View_name 


SET 


column name 


expression 


表 27.8 UPDATE 语句 的 参数 说 明 
说 明 

需要 更 新 的 表 的 名 称 。 如 果 该 表 不 在 当前 服务 器 或 数据 库 中 ， 或 不 为 当前 用 户 所 有 ， 这 个 名 称 
可 用 链接 服务 器 、 数 据 库 和 所 有 者 名 称 来 限定 
要 更 新 的 视图 的 名 称 。 通 过 view_name 来 引用 的 视图 必须 是 可 更 新 的 。 用 UPDATE 语句 进行 
的 修改 ， 至 多 只 能 影响 视图 的 FROM 子 句 所 引用 的 基 表 中 的 一 个 
指定 要 更 新 的 列 或 变量 名 称 的 列表 
含有 要 更 改 数据 的 列 的 名 称 ,column_name 必须 驻 留 于 UPDATE 子 句 中 所 指定 的 表 或 视图 中 。 
标识 列 不 能 进行 更 新 。 如 果 指 定 了 限定 的 列 名 称 ， 限 定 符 必须 同 UPDATE 子 句 中 的 表 或 视图 
的 名 称 相 匹配 
变量 、 字 面值 、 表 达 式 或 加 上 括号 返回 单个 值 的 subSELECT 语句 。expression 返回 的 值 将 替 


换 column_ name 或 @variable 中 的 现 有 值 


WHERE 


<search condition> 


指定 条 件 来 限定 所 更 新 的 行 。 根 据 所 使 用 的 WHERE 子 句 的 形式 ， 有 两 种 更 新 形式 : 

搜索 更 新 指定 搜索 条 件 来 限定 要 删除 的 行 ; 

定位 更 新 使 用 CURRENT OF 子 句 指定 游标 。 更 新 操作 发 生 在 游标 的 当前 位 置 

为 要 更 新 行 指定 需 满足 的 条 件 。 搜 索 条件 也 可 以 是 连接 所 基于 的 条 件 。 对 搜索 条 件 中 可 以 包含 
的 谓词 数量 没有 限制 


多 wt 起 一 定 要 确保 不 要 忽略 WHERE 子 句 ， 除 非 想 要 更 新 表 中 的 所 有 行 。 


下 面 应 用 UPDATE 语句 将 指定 职员 的 工资 进行 调整 ， 例 如 ， 将 小 璇 的 基本 工资 由 2600 元 修改 为 3000 
元 。 其 SQL 语句 如 下 : 

update tab_laborage set jbgz=3000 where name=' 小 璇 ' 

2. DELETE 语句 


DELETE 语句 实现 删除 数据 记录 。DELETE 语句 的 语法 如 下 : 
DELETE FROM <table_name > 
[WHERE<search condition>] 


DELETE 语句 的 参数 说 明 如 表 27.9 所 示 。 
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27.10.2 ”数据 的 更 新 和 删除 技术 


付费 信息 管理 页 面 在 实现 信息 审核 及 删除 的 功能 时 应 用 到 了 UPDATE 更 新 语句 和 DELETE 删除 语句 。 
下 面 对 这 两 个 语句 进行 详细 地 讲解 。 
1. UPDATE 语句 
UPDATE 语句 用 来 改变 单行 上 的 一 列 或 多 列 的 值 ， 或 者 改变 单个 表 中 选 定 的 一 些 行 上 的 多 个 列 值 。 
UPDATE 语句 的 语法 如 下 : 
UPDATE<table_name | view_name> 
SET <column_name>=<expression> 
[.…,<last column_name>=<last expression>] 
[WHERE<search_condition>] 
UPDATE 语句 的 参数 说 明 如 表 27.8 所 示 。 


参数 


table_ name 


View_name 


SET 


column name 


expression 


表 27.8 UPDATE 语句 的 参数 说 明 
说 明 

需要 更 新 的 表 的 名 称 。 如 果 该 表 不 在 当前 服务 器 或 数据 库 中 ， 或 不 为 当前 用 户 所 有 ， 这 个 名 称 
可 用 链接 服务 器 、 数 据 库 和 所 有 者 名 称 来 限定 
要 更 新 的 视图 的 名 称 。 通 过 view_name 来 引用 的 视图 必须 是 可 更 新 的 。 用 UPDATE 语句 进行 
的 修改 ， 至 多 只 能 影响 视图 的 FROM 子 句 所 引用 的 基 表 中 的 一 个 
指定 要 更 新 的 列 或 变量 名 称 的 列表 
含有 要 更 改 数据 的 列 的 名 称 ,column_name 必须 驻 留 于 UPDATE 子 句 中 所 指定 的 表 或 视图 中 。 
标识 列 不 能 进行 更 新 。 如 果 指 定 了 限定 的 列 名 称 ， 限 定 符 必须 同 UPDATE 子 句 中 的 表 或 视图 
的 名 称 相 匹配 
变量 、 字 面值 、 表 达 式 或 加 上 括号 返回 单个 值 的 subSELECT 语句 。expression 返回 的 值 将 替 


换 column_ name 或 @variable 中 的 现 有 值 


WHERE 


<search condition> 


指定 条 件 来 限定 所 更 新 的 行 。 根 据 所 使 用 的 WHERE 子 句 的 形式 ， 有 两 种 更 新 形式 : 

搜索 更 新 指定 搜索 条 件 来 限定 要 删除 的 行 ; 

定位 更 新 使 用 CURRENT OF 子 句 指定 游标 。 更 新 操作 发 生 在 游标 的 当前 位 置 

为 要 更 新 行 指定 需 满足 的 条 件 。 搜 索 条件 也 可 以 是 连接 所 基于 的 条 件 。 对 搜索 条 件 中 可 以 包含 
的 谓词 数量 没有 限制 


多 wt 起 一 定 要 确保 不 要 忽略 WHERE 子 句 ， 除 非 想 要 更 新 表 中 的 所 有 行 。 


下 面 应 用 UPDATE 语句 将 指定 职员 的 工资 进行 调整 ， 例 如 ， 将 小 璇 的 基本 工资 由 2600 元 修改 为 3000 
元 。 其 SQL 语句 如 下 : 

update tab_laborage set jbgz=3000 where name=' 小 璇 ' 

2. DELETE 语句 


DELETE 语句 实现 删除 数据 记录 。DELETE 语句 的 语法 如 下 : 
DELETE FROM <table_name > 
[WHERE<search condition>] 


DELETE 语句 的 参数 说 明 如 表 27.9 所 示 。 
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表 27.9 DELETE 语句 的 参数 说 明 


参数 说 有明 
可 选 的 关键 字 ， 可 用 在 DELETE 关键 字 与 目标 table_ name、view_name 或 rowset function_ 
EROM es 
limited 之 间 
table name 是 要 从 其 中 删除 行 的 表 的 名 称 


<search condition> | 指定 删除 行 的 限定 条 件 。 对 搜索 条 件 中 可 以 包含 的 谓词 数量 没有 限制 


下 面 应 用 DELETE 语句 删除 “职员 姓名 = 小 璇 ”的 员工 基本 信息 。 其 SQL 语句 如 下 : 
DELETE tab_staffer WHERE ygname=' 小 璇 ' 


27.10.3 ”付费 信息 显示 的 实现 过 程 


管理 员 在 后 台 功 能 导航 区 的 付费 信息 显示 方式 栏 中 选择 相应 的 信息 类 别 ， 然 后 按 “ 已 付费 ”、“ 未 付 
费 ” 或 “全 部 ”中 的 任意 一 种 状态 对 付费 信息 进行 管理 。 例 如 ， 在 “信息 类 别 ” 下 拉 列 表 框 中 选择 “公寓 
信息 ”， 在 付费 状态 中 选中 “全 选 ” 单 选 按钮 ， 单 击 “ 检 索 ” 按 钮 提交 表单 ， 程 序 将 按 指定 条 件 显 示 出 符 
合 条 件 的 所 有 信息 ， 运 行 结 果 如 图 27.31 所 示 。 
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;条 条 不 4 条 第 1 页/ 共 1 页 1 


27.31 付费 信息 显示 页 面 的 运行 结果 


本 系统 提供 了 一 组 单 选 按钮 组 成 的 “付费 状态 ”选项 组 ， 付 费 状态 分 为 已 付费 、 未 付费 和 全 部 3 个 选 
项 。 选 中 “未 付费 ” 单 选 按钮 ， 则 传递 的 值 为 “0”; 选中 “已 付费 ” 单 选 按钮 ， 则 传递 的 值 为 “1”; 选 
中 “全 部 ” 单 选 按钮 ， 则 传递 的 值 为 “all”。 还 提供 了 一 个 下 拉 列 表 框 ， 供 用 户 选择 信息 类 别 。 将 这 些 单 
选 按钮 与 下 拉 列 表 框 都 在 一 个 表单 中 实现 ， 这 样 ， 当 单 击 “ 检 索 ” 按 钮 提交 表单 后 ， 选 择 的 状态 会 通过 表 
单 进行 传递 。 其 表单 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance27\admin\left.php) 

@ <formname="form3" method="post" action="find_mianfei.php" target="mainFrame"> 

<tr> 

<td height="65" align="center"> 


局 
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@ <feldset style="height:60:width:210"> 
<legend> 友 审核 状态 </legend> 
<input name="state" type="radio" class="input1" value="1"> 已 审核 
<input name="state" type="radio" class="input1" value="0" checked> 未 审核 
<input name="state" type="radio" class="input1" value="all"> 全 部 
</fieldset> 
</td> 


<td height="34" align="center" background="images/info_d.gif"> 信 息 类 别 : 
<select name="type"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 </option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 </option> 


</select> 
<input type="submit" name="Submit" value=" 检 索 "> 


yg 
Ds 说 明 @ target: 指定 链接 的 目标 窗口 ，mainFrame 为 内 容 显 示 区 的 框架 名 称 。 另 外 ，target 有 4 个 选 
项 值 。 
_blank: 指定 将 链接 的 目标 文件 加 载 到 未 命名 的 新 浏览 器 窗口 中 。 
_parent: 指定 将 链接 的 目标 文件 加 载 到 包含 链接 的 父 框 架 页 或 窗口 中 ， 如 果 包 含 链接 的 框 不 是 
许 套 的 ， 则 链接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 。 
_self: 指定 将 链接 的 目标 文件 加 载 到 链接 所 在 的 同一 框架 或 窗口 中 。 
_top: 指定 将 链接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 ， 并 由 此 删除 所 有 框架 。 
@ <fieldset><legend>...</legend></fieldset> 标 签 : 在 字符 集 包 含 的 文本 和 其 他 元 素 外 面 绘制 一 个 方 
框 。 该 元 素 是 块 元 素 ， 必 须 成 对 出 现 。 需 要 注意 的 是 ，fieldset 必须 用 在 form 表单 里 ， 一 个 表单 可 以 有 
多 个 <fieldset>...</fieldset>， 每 对 <fieldset></fieldset> 为 一 组 ， 每 组 的 内 容 描述 使 用 <legend> 设 置 标题 
名 称 。 


提交 表单 信息 到 find_mianfei.php 页 ， 程 序 将 按 管理 员 选 择 的 指定 条 件 显示 出 符合 条 件 的 所 有 信息 。 如 
果 管 理 员 选中 “全 部 ” 单 选 按钮 ， 那 么 系统 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance27\admin\find_fufei.php) 


<?php 

Include("../conn/conn.php"); // 连 接 数据 库 文件 
$state=$_POSTIpayfor]; /获取 管理 员 选 择 的 付费 状态 
Stype=$_POST[select]; // 获 取 管理 员 选 择 的 信息 类 别 


// 如 果 管 理 员 选 择 的 付费 状态 为 “全 部 ”， 则 查询 管理 员 指 定 信息 类 别 下 的 所 有 付费 信息 ， 并 按 id 降序 排列 
if($state=="all"}{ 

$sql=mysql_query("select * from tb_leaguerinfo where type='$type' order by id"); 
} 


else{ 
// 如 果 管理 员 选 择 的 付费 状态 为 “已 付费 ”或 “未 付费 ”状态 ， 则 按 管理 员 指 定 的 条 件 进行 查询 ， 并 按 id 降序 排列 
$sql=mysql_query("select* from tb_leaguerinfo where type='$type' and checkstate=$state order by id"); 


全 


上 
$info=mysql_fetch_array($sql); /执行 SQL 语句 
?> 
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@ <feldset style="height:60:width:210"> 
<legend> 友 审核 状态 </legend> 
<input name="state" type="radio" class="input1" value="1"> 已 审核 
<input name="state" type="radio" class="input1" value="0" checked> 未 审核 
<input name="state" type="radio" class="input1" value="all"> 全 部 
</fieldset> 
</td> 


<td height="34" align="center" background="images/info_d.gif"> 信 息 类 别 : 
<select name="type"> 
<option value=" 招 聘 信息 ">- 招 聘 信息 </option> 
<option value=" 求 职 信息 " selected>- 求 职 信息 </option> 


</select> 
<input type="submit" name="Submit" value=" 检 索 "> 


yg 
Ds 说 明 @ target: 指定 链接 的 目标 窗口 ，mainFrame 为 内 容 显 示 区 的 框架 名 称 。 另 外 ，target 有 4 个 选 
项 值 。 
_blank: 指定 将 链接 的 目标 文件 加 载 到 未 命名 的 新 浏览 器 窗口 中 。 
_parent: 指定 将 链接 的 目标 文件 加 载 到 包含 链接 的 父 框 架 页 或 窗口 中 ， 如 果 包 含 链接 的 框 不 是 
许 套 的 ， 则 链接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 。 
_self: 指定 将 链接 的 目标 文件 加 载 到 链接 所 在 的 同一 框架 或 窗口 中 。 
_top: 指定 将 链接 的 目标 文件 加 载 到 整个 浏览 器 窗口 中 ， 并 由 此 删除 所 有 框架 。 
@ <fieldset><legend>...</legend></fieldset> 标 签 : 在 字符 集 包 含 的 文本 和 其 他 元 素 外 面 绘制 一 个 方 
框 。 该 元 素 是 块 元 素 ， 必 须 成 对 出 现 。 需 要 注意 的 是 ，fieldset 必须 用 在 form 表单 里 ， 一 个 表单 可 以 有 
多 个 <fieldset>...</fieldset>， 每 对 <fieldset></fieldset> 为 一 组 ， 每 组 的 内 容 描述 使 用 <legend> 设 置 标题 
名 称 。 


提交 表单 信息 到 find_mianfei.php 页 ， 程 序 将 按 管理 员 选 择 的 指定 条 件 显示 出 符合 条 件 的 所 有 信息 。 如 
果 管 理 员 选中 “全 部 ” 单 选 按钮 ， 那 么 系统 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance27\admin\find_fufei.php) 


<?php 

Include("../conn/conn.php"); // 连 接 数据 库 文件 
$state=$_POSTIpayfor]; /获取 管理 员 选 择 的 付费 状态 
Stype=$_POST[select]; // 获 取 管理 员 选 择 的 信息 类 别 


// 如 果 管 理 员 选 择 的 付费 状态 为 “全 部 ”， 则 查询 管理 员 指 定 信息 类 别 下 的 所 有 付费 信息 ， 并 按 id 降序 排列 
if($state=="all"}{ 

$sql=mysql_query("select * from tb_leaguerinfo where type='$type' order by id"); 
} 


else{ 
// 如 果 管理 员 选 择 的 付费 状态 为 “已 付费 ”或 “未 付费 ”状态 ， 则 按 管理 员 指 定 的 条 件 进行 查询 ， 并 按 id 降序 排列 
$sql=mysql_query("select* from tb_leaguerinfo where type='$type' and checkstate=$state order by id"); 


全 


上 
$info=mysql_fetch_array($sql); /执行 SQL 语句 
?> 
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… // 省 略 供求 信息 标题 的 HTML 代码 部 分 
<?php 
if($infoX{ // 如 果 检 索 到 了 查询 记录 
dof /应 用 do…while 循环 语句 输出 付费 信息 
if($info[checkstate]==1){ /如果 付 费 状 态 的 值 为 1 
$state1=" 已 付费 "; // 则 将 “已 付费 ” 赋 给 变量 $state1 
Jelse{ // 如 果 付 费 状 态 的 值 为 0 
$state1=" 未 付费 "; // 则 将 “未 付费 ” 赋 给 变量 $state1 
?> 
< 上 一 一 一 一 一 一 一 一 -输出 指定 符合 查询 条 件 的 付费 信息 一 一 一 一 一 一 一 一 一 一 
<tr bgcolor="#FFFFFF"> 


<td>&nbsp;<?php echo $infoltitle];?></td> 

<td width="204">&nbsp;<?php echo S$info[content];?></td> 

<td>&nbsp;<?php echo $infollinkman];?></td> 

<td>&nbsp;<?php echo Sinfo[tel];?></td> 

<td>&nbsp;<?php echo $info[sdate];?></td> 

<td>&nbsp;<?php echo $info[showday];?></td> 

<td align="center" class="style11"><?php echo $state1;?></td> 

<td align="center" bgcolor="#FFFFFF"> 
<a href="statefu_ok.php?id=<?php echo Sinfolid];?>&type=<?php echo $type;?>&state=<?php echo 
$state;?>" > 审核 </a> 
/<a href="fudel_ok.php?id=<?php echo Sinfo[id];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
删除 </a> 
</td> 
</tr> 
和 
<?php 

}while($info=mysql_fetch_array($sql)); /ldo…while 循环 语句 结束 

} /所 条 件 语句 结束 


// 如 果 未 检索 到 与 管理 员 指 定 条 件 相 匹 配 的 记录 ， 则 输出 相关 的 提示 信息 
else{ 


?> 
<tr align="center" bgcolor="#FFFFFF"> 
<td colspan="8"> 对 不 起 ， 您 检索 的 信息 不 存在 ! </td> 
</tr> 
<?php 


?> 
SC 多 
和 上 面 代 码 中 省 略 了 分 页 显示 代码 的 部 分 ， 限 于 篇 幅 ， 参 见 本 书 附 赠 的 光盘 。 


27.10.4 ”付费 信息 审核 的 实现 过 程 


经 过 审核 的 信息 说 明 该 信息 为 已 付款 信息 。 如 果 企 业 或 个 人 用 户 已 登录 供求 信息 但 未 直接 付费 ， 需 要 
后 期 付款 ， 那 么 必须 进行 付款 ， 同 时 管理 员 进 行 审核 ， 经 过 审核 的 信息 即 可 在 前 台 进 行 显示 。“ 审 核 ” 超 
链接 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance27\admin\find_fufei.php) 
<a href="statefu_ok.php?id=<?php echo Sinfolid];?>&type=<?php echo S$type;?>&state=<?php echo 
$state;?>" > 审核 </a> 


© 
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… // 省 略 供求 信息 标题 的 HTML 代码 部 分 
<?php 
if($infoX{ // 如 果 检 索 到 了 查询 记录 
dof /应 用 do…while 循环 语句 输出 付费 信息 
if($info[checkstate]==1){ /如果 付 费 状 态 的 值 为 1 
$state1=" 已 付费 "; // 则 将 “已 付费 ” 赋 给 变量 $state1 
Jelse{ // 如 果 付 费 状 态 的 值 为 0 
$state1=" 未 付费 "; // 则 将 “未 付费 ” 赋 给 变量 $state1 
?> 
< 上 一 一 一 一 一 一 一 一 -输出 指定 符合 查询 条 件 的 付费 信息 一 一 一 一 一 一 一 一 一 一 
<tr bgcolor="#FFFFFF"> 


<td>&nbsp;<?php echo $infoltitle];?></td> 

<td width="204">&nbsp;<?php echo S$info[content];?></td> 

<td>&nbsp;<?php echo $infollinkman];?></td> 

<td>&nbsp;<?php echo Sinfo[tel];?></td> 

<td>&nbsp;<?php echo $info[sdate];?></td> 

<td>&nbsp;<?php echo $info[showday];?></td> 

<td align="center" class="style11"><?php echo $state1;?></td> 

<td align="center" bgcolor="#FFFFFF"> 
<a href="statefu_ok.php?id=<?php echo Sinfolid];?>&type=<?php echo $type;?>&state=<?php echo 
$state;?>" > 审核 </a> 
/<a href="fudel_ok.php?id=<?php echo Sinfo[id];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
删除 </a> 
</td> 
</tr> 
和 
<?php 

}while($info=mysql_fetch_array($sql)); /ldo…while 循环 语句 结束 

} /所 条 件 语句 结束 


// 如 果 未 检索 到 与 管理 员 指 定 条 件 相 匹 配 的 记录 ， 则 输出 相关 的 提示 信息 
else{ 


?> 
<tr align="center" bgcolor="#FFFFFF"> 
<td colspan="8"> 对 不 起 ， 您 检索 的 信息 不 存在 ! </td> 
</tr> 
<?php 


?> 
SC 多 
和 上 面 代 码 中 省 略 了 分 页 显示 代码 的 部 分 ， 限 于 篇 幅 ， 参 见 本 书 附 赠 的 光盘 。 


27.10.4 ”付费 信息 审核 的 实现 过 程 


经 过 审核 的 信息 说 明 该 信息 为 已 付款 信息 。 如 果 企 业 或 个 人 用 户 已 登录 供求 信息 但 未 直接 付费 ， 需 要 
后 期 付款 ， 那 么 必须 进行 付款 ， 同 时 管理 员 进 行 审核 ， 经 过 审核 的 信息 即 可 在 前 台 进 行 显示 。“ 审 核 ” 超 
链接 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance27\admin\find_fufei.php) 
<a href="statefu_ok.php?id=<?php echo Sinfolid];?>&type=<?php echo S$type;?>&state=<?php echo 
$state;?>" > 审核 </a> 


© 
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管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传递 到 
数据 处 理 页 statefu_ok.php， UPDATE 语句 将 付费 状态 设置 为 1， 说 明 该 信息 已 经 付款 。 数 据 处 理 页 的 代 
码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance27\admin\statefu_ok.php) 


<?phl 

让 .php"); // 连 接 数据 库 文件 
$id=$_GET[id]; // 获 取信 息 id 的 值 
Stype=$_GET[type]; // 获 取信 息 类 型 
$state=$_GET[state]; /获取 信息 付费 状态 


@ $sql=mysql_query("update tb_leaguerinfo set checkstate=1 where id=$id"); 
// 更 新 对 应 付费 信息 的 状态 为 已 付款 
if($sql}{ // 如 果 更 新 操作 成 功 
@ ”echo "<script>alert(' 该 信息 已 经 通过 审核 ! 
小 window.location.href='find_fufei.php?type=$type&state=$state';</script>"; ”// 弹 出 操作 成 功 提示 信息 


} 

else{ /| 如果 更 新 操作 失败 
echo "<script>alert(' 该 信息 审核 操作 失败 ! ');history.back);</script>"; // 弹 出 操作 失败 信息 

) 


> 


CA 
@ update.…set: 用 来 修改 指定 表 中 的 数据 。 

@ type=$type&state=$state: 将 变量 type 与 state 的 值 重 新 传 到 find fufeiphp 页 ， 目 的 是 使 
find_fufei.php 页 中 的 $type 和 Sstate 变量 重新 获得 信息 类 型 和 付费 状态 值 (管理 员 选 择 的 检索 条 件 )， 从 
而 返回 到 付费 信息 管理 页 ， 并 更 新 数据 信息 。 如 果 数 据 处 理 页 不 对 这 两 个 变量 进行 传 值 ， 那 么 在 返回 到 
付费 信息 管理 页 find fufeiphp 页 时 将 会 因为 检索 不 到 变量 的 值 而 出 错 。 


27.10.5 ”付费 信息 删除 的 实现 过 程 


付费 信息 管理 页 中 “删除 ” 超 链 接 的 代码 如 下 : 
《代码 位 置 ， 光盘 \TMNVInstance\27\adminfind_fufeiphp) 
te Eu echo S$infolid];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传递 到 
数据 处 理 页 fadel_ ok.php， 用 DELETE 语句 将 id 指定 的 供求 信息 删除 。 数 据 处 理 页 的 代码 如 下 : 
《代码 位 置 ， 光盘 \TMNInstance\27\admin\fudel ok-php) 


<?php 

include("../conn/conn.php"); // 连 接 数据 库 文 件 

$id=$_GET[id]; // 获 取信 息 id 的 值 

Stype=$_GET[type]; // 获 取信 息 类 型 

S$state=$_GETI[state]; // 获 取信 息 付费 状态 

$sql=mysql_query("delete from tb_leaguerinfo where id=$id"); /删除 对 应 的 供求 信息 

if($sql){ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 
echo "<script>alert(' 该 信息 已 经 删除 ! ');window.location.href='find_fufei.php?type=$type&state=$state'; 

</script>"; 

else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 信息 


_ 回 ) 
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管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传递 到 
数据 处 理 页 statefu_ok.php， UPDATE 语句 将 付费 状态 设置 为 1， 说 明 该 信息 已 经 付款 。 数 据 处 理 页 的 代 
码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance27\admin\statefu_ok.php) 


<?phl 

让 .php"); // 连 接 数据 库 文件 
$id=$_GET[id]; // 获 取信 息 id 的 值 
Stype=$_GET[type]; // 获 取信 息 类 型 
$state=$_GET[state]; /获取 信息 付费 状态 


@ $sql=mysql_query("update tb_leaguerinfo set checkstate=1 where id=$id"); 
// 更 新 对 应 付费 信息 的 状态 为 已 付款 
if($sql}{ // 如 果 更 新 操作 成 功 
@ ”echo "<script>alert(' 该 信息 已 经 通过 审核 ! 
小 window.location.href='find_fufei.php?type=$type&state=$state';</script>"; ”// 弹 出 操作 成 功 提示 信息 


} 

else{ /| 如果 更 新 操作 失败 
echo "<script>alert(' 该 信息 审核 操作 失败 ! ');history.back);</script>"; // 弹 出 操作 失败 信息 

) 


> 


CA 
@ update.…set: 用 来 修改 指定 表 中 的 数据 。 

@ type=$type&state=$state: 将 变量 type 与 state 的 值 重 新 传 到 find fufeiphp 页 ， 目 的 是 使 
find_fufei.php 页 中 的 $type 和 Sstate 变量 重新 获得 信息 类 型 和 付费 状态 值 (管理 员 选 择 的 检索 条 件 )， 从 
而 返回 到 付费 信息 管理 页 ， 并 更 新 数据 信息 。 如 果 数 据 处 理 页 不 对 这 两 个 变量 进行 传 值 ， 那 么 在 返回 到 
付费 信息 管理 页 find fufeiphp 页 时 将 会 因为 检索 不 到 变量 的 值 而 出 错 。 


27.10.5 ”付费 信息 删除 的 实现 过 程 


付费 信息 管理 页 中 “删除 ” 超 链 接 的 代码 如 下 : 
《代码 位 置 ， 光盘 \TMNVInstance\27\adminfind_fufeiphp) 
te Eu echo S$infolid];?>&type=<?php echo $type;?>&state=<?php echo $state;?>"> 
管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传递 到 
数据 处 理 页 fadel_ ok.php， 用 DELETE 语句 将 id 指定 的 供求 信息 删除 。 数 据 处 理 页 的 代码 如 下 : 
《代码 位 置 ， 光盘 \TMNInstance\27\admin\fudel ok-php) 


<?php 

include("../conn/conn.php"); // 连 接 数据 库 文 件 

$id=$_GET[id]; // 获 取信 息 id 的 值 

Stype=$_GET[type]; // 获 取信 息 类 型 

S$state=$_GETI[state]; // 获 取信 息 付费 状态 

$sql=mysql_query("delete from tb_leaguerinfo where id=$id"); /删除 对 应 的 供求 信息 

if($sql){ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 
echo "<script>alert(' 该 信息 已 经 删除 ! ');window.location.href='find_fufei.php?type=$type&state=$state'; 

</script>"; 

else{ // 如 果 删 除 操作 失败 ， 则 弹出 提示 信息 


_ 回 ) 
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echo "<script>alert( 该 信息 删除 操作 失败 ! ');history.back();</script>"; 
册 


2 


人 注意 在 删除 数据 后 ， 仍 需要 将 变量 $type 和 $state 的 值 重新 传递 给 付费 信息 管理 页 find fnfeiphp， 
目的 是 返回 到 付费 信息 管理 页 ， 查 看 执行 删除 操作 后 的 状态 。 


27.10.6 单元 测试 


在 开发 完 管理 员 模块 后 ， 对 该 模块 进行 单元 测试 。 当 管理 员 对 付费 供求 信息 进行 审核 后 ， 审 核 操作 成 
功 了 ， 但 却 弹出 如 图 27.32 所 示 的 错误 提示 。 


ES 

SS 同 本 本 下 号 
ee es 

口 已 械 个 未 宙 硕 个 全 部 四 修 现 在 的 位 团 ; 九 九 朗 供 玉 全 忆 同 》 后 党 理 系统 


my: RE 网 | 
TT 后 了。 和 9 区。 梳 N 

es MT , SM TE 

Ea 人 向 册 人 至 和 | 信息 夫 别 为 空 ， 说 明 役 有 获取 到 什 


Cd [ES 到 | 
图 27.32 审核 付费 供求 信息 的 错误 提示 


在 图 27.32 中 的 错误 提示 中 可 以 看 出 ,在 付费 信息 管理 页 的 第 16 行 和 第 37 行 出 现 问 题 。 下 面 看 一 下 出 
现 问题 的 这 两 行 代码 : 


昌 多 和 人 应 骂 示 万 式 


if($state=="all"}{ // 如 果 管理 员 选 择 的 付费 状态 为 “全 部 ”， 则 执行 下 面 的 SQL 语句 
$sql1=mysql_query("select count(*) as total from tb_leaguerinfo ”where type='$type' order by id"); 
jelsef /否则 ， 执 行 下 面 的 SQL 语句 


$sql1=mysql_query("select count(*) as total from tb_leaguerinfo where type='$type' and checkstate= 
$state order by id"); 
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$minfo=mysql_fetch_array($sql1); 


Pes ‘re 行 代码 nn 
S$total=$minfo[total]; 
从 代码 中 可 以 看 出 , SQL 语句 的 书写 并 没有 错误 根据 图 27.32 所 示 的 页 面 的 结果 , 当前 信息 类 别 为 空 ， 
则 说 明 管 理 员 选 择 的 信息 类 别 没有 传 过 来 值 ， 由 此 可 以 看 出 ， 这 是 由 于 在 执行 审核 后 页 面 重新 剧 新 了 ， 因 
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<a href="statefu_ok.php?id=<?php echo $infofid];?>"> 审 核 </a> 
审核 处 理 页 的 源 代码 如 下 : 
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<?phl 
Sid-S GETtd /获取 信息 id 的 值 
$sql=mysql_query("update tb_leaguerinfo set checkstate=1 where id=$id"); 


@ 
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// 更 新 对 应 付费 信息 的 状态 为 已 付款 
if($sql}{ // 如 果 更 新 操作 成 功 ， 弹 出 提示 信息 
echo "<script>alert(' 该 信息 已 经 通过 审核 ! ');window.location.href='find_fufei.php;</script>"; 

解决 该 问题 的 方法 需要 在 “审核 ” 超 链接 传 值 时 将 管理 员 选 定 的 “信息 类 型 ”和 “审核 状态 ”的 变量 
值 一 同 传递 到 数据 处 理 页 ， 当 审核 操作 完成 后 ， 再 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 重 新 传递 给 付 
费 信 息 管理 页 find_fufei.php。 

“审核 ” 超 链接 修改 后 的 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance27\admin\find_fufei.php) 

<a href="statefu_ok.php?id=<?php echo Sinfo[lid];?>&type=<?php echo $type;?>&state=<?php echo 

$state;?>" > 审核 </a> 


| 


管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传递 到 
数据 处 理 页 statefu_ok.php， 在 执行 完 更 新 操作 后 ， 需 要 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 重 新 传递 
到 付费 信息 管理 页 fnd fufeiphp， 代 码 如 下 。 

(代码 位 置 ， 光 盘 \TM\Instance\27\admin\statefu_ok.php) 


<?php 
$id=$_GETI[id]; // 获 取信 息 id 的 值 
$type=$_GET[type]; /获取 信息 类 型 
$state=$_GETI[state]; // 获 取信 息 付 费 状 态 

$sql=mysql_query("update tb_leaguerinfo set checkstate=16 where id=$id"); 

// 更 新 对 应 付费 信息 的 状态 为 已 付款 

if($sqX{ 1/ 如果 更 新 操作 成 功 

echo "<script>alert(' 该 信息 已 经 通过 审核 ! 
");window.location.href="find_fufei.php?type=$type&state=$state';</script>"; // 弹 出 操作 成 功 提示 信息 
} 


27.11 站 结 


本 章 依据 软件 开发 流程 介绍 了 易 查 供求 信息 网 的 开发 过 程 。 在 开发 任何 一 个 项 目前 ， 首 先 要 充分 做 好 
前 期 准备 ， 如 完善 的 需求 分 析 、 清 晰 的 业务 流程 、 合 理 的 程序 结构 等 ， 这 样 ， 在 后 期 的 程序 开发 中 才 会 得 
心 应 手 ， 有 备 无 患 。 
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// 更 新 对 应 付费 信息 的 状态 为 已 付款 
if($sql}{ // 如 果 更 新 操作 成 功 ， 弹 出 提示 信息 
echo "<script>alert(' 该 信息 已 经 通过 审核 ! ');window.location.href='find_fufei.php;</script>"; 

解决 该 问题 的 方法 需要 在 “审核 ” 超 链接 传 值 时 将 管理 员 选 定 的 “信息 类 型 ”和 “审核 状态 ”的 变量 
值 一 同 传递 到 数据 处 理 页 ， 当 审核 操作 完成 后 ， 再 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 重 新 传递 给 付 
费 信 息 管理 页 find_fufei.php。 

“审核 ” 超 链接 修改 后 的 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance27\admin\find_fufei.php) 

<a href="statefu_ok.php?id=<?php echo Sinfo[lid];?>&type=<?php echo $type;?>&state=<?php echo 

$state;?>" > 审核 </a> 


| 


管理 员 单 击 对 应 主题 信息 后 面 的 “审核 ” 超 链接 ， 将 信息 所 对 应 的 id 值 、 信 息 类 型 及 审核 状态 传递 到 
数据 处 理 页 statefu_ok.php， 在 执行 完 更 新 操作 后 ， 需 要 将 “信息 类 型 ”和 “审核 状态 ”的 变量 值 重 新 传递 
到 付费 信息 管理 页 fnd fufeiphp， 代 码 如 下 。 

(代码 位 置 ， 光 盘 \TM\Instance\27\admin\statefu_ok.php) 


<?php 
$id=$_GETI[id]; // 获 取信 息 id 的 值 
$type=$_GET[type]; /获取 信息 类 型 
$state=$_GETI[state]; // 获 取信 息 付 费 状 态 

$sql=mysql_query("update tb_leaguerinfo set checkstate=16 where id=$id"); 

// 更 新 对 应 付费 信息 的 状态 为 已 付款 

if($sqX{ 1/ 如果 更 新 操作 成 功 

echo "<script>alert(' 该 信息 已 经 通过 审核 ! 
");window.location.href="find_fufei.php?type=$type&state=$state';</script>"; // 弹 出 操作 成 功 提示 信息 
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本 章 依据 软件 开发 流程 介绍 了 易 查 供求 信息 网 的 开发 过 程 。 在 开发 任何 一 个 项 目前 ， 首 先 要 充分 做 好 
前 期 准备 ， 如 完善 的 需求 分 析 、 清 晰 的 业务 流程 、 合 理 的 程序 结构 等 ， 这 样 ， 在 后 期 的 程序 开发 中 才 会 得 
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图 书馆 管理 系统 


(名! 视频 讲解 : 137 分 钟 ) 


随 着 网 络 技 术 的 高 速 发 展 和 计算 机 应 用 的 普及 ， 利 用 计算 机 对 图 
书馆 的 日 常 工作 进行 管理 势 在 必 行 。 虽然 目前 很 多 大 型 图 书馆 已 经 有 
一 整套 比较 完善 的 管理 系统 ， 但 是 在 一 些 中 小 型 图 书馆 中 ， 大 部 分 工 
作 仍 需 由 手工 完成 ， 工 作 效率 低 ， 管 理 员 不 能 及 时 了 解 馆 内 各 类 图 书 
的 借阅 情况 ， 读 者 需要 的 图 书 难以 在 短 时 间 内 找到 ， 不 便于 动态 、 及 
时 地 调整 图 书 结构 。 为 了 更 好 地 适应 当前 读者 的 借阅 需求 ， 解 决 手工 
管理 中 存在 的 许多 问题 ， 越 来 越 多 的 中 小 型 图 书馆 正在 逐步 向 计算 机 
信息 化 管理 转变 。 本 章 通过 开发 一 个 图 书馆 管理 系统 ， 为 读者 详细 讲 
解 项 目 开发 流程 。 

通过 阅读 本 章 内 容 ， 你 可 以 : 

MH 了 解 图 书馆 管理 系统 开发 的 基本 过 程 

MW 掌握 系统 设计 的 方法 

Wm 掌握 如 何 分 析 并 设计 数据 库 、 数 据 表 

Mi 掌握 多 表 查 询 的 方法 

Nm 了 解 主 要 功能 模块 的 实现 方法 

M 掌握 如 何 自动 计算 图 书 娄 还 日 期 

由 掌握 内 联接 和 外 联接 语句 的 使 用 方法 
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28.1 图 书馆 管理 系统 概述 


知 源 图 书馆 是 师范 学 校 的 大 型 图 书馆 。 随 着 图 书馆 规模 的 不 断 扩 大 ， 图 书 品种 和 数量 也 逐渐 增多 。 但 
同时 ， 图 书馆 采用 的 传统 的 人 工 管理 方式 暴露 了 一 些 问 题 。 例 如 ， 查 找 读者 借阅 的 某 一 本 图 书 的 具体 摆 放 
位 置 ， 需 要 靠 人 工 记忆 在 书 海中 苦 苦 查找 ， 由 于 图 书 存储 量 大 ， 很 难 准确 定位 图 书 的 具体 位 置 ， 因 此 每 天 
都 要 浪费 大 量 宝贵 的 时 间 。 学 校 图 书馆 为 提高 工作 效率 ， 同 时 摆脱 人 工 管理 中 出 现 的 种 种 弊端 ， 现 委托 某 
单位 开发 一 个 学 校 图 书馆 管理 系统 。 


28.2 需求 分 析 


通过 计算 机 对 图 书 进 行 管 理 ， 不 仅 为 图 书馆 的 管理 注入 了 新 的 生机 ， 而 且 在 运营 过 程 中 节省 了 大 量 的 
人 力 、 物 力 、 财 力 和 时 间 ， 可 以 提高 图 书馆 的 工作 效率 ， 还 为 图 书馆 在 读者 群 中 树立 一 个 全 新 的 形象 ， 为 
图 书馆 日 后 发 展 葛 定 一 个 良好 的 基础 。 通 过 对 一 些 大 型 图 书馆 的 实际 考察 、 分 析 ， 并 结合 图 书馆 的 要 求 以 
及 实际 的 市 场 调查 ， 要 求 本 系统 具有 以 下 功能 。 
网 站 设计 页 面 要 求 美观 大 方 、 个 性 化 ， 功 能 全 面 ， 操 作 简单 。 
要 求实 现 基 础 信息 的 管理 平台 。 
要 求 对 所 有 读者 进行 管理 。 
要 求实 现 图 书 借阅 排行 ， 了 解 当前 的 畅销 书 。 
商品 分 类 详尽 ， 可 按 不 同类 别 查 看 图 书信 息 。 
提供 快速 的 图 书信 息 、 图 书 借阅 检索 功能 ， 保 证 数据 查询 的 灵活 性 。 
实现 图 书 借阅 、 图 书 续 借 、 图 书 归还 的 功能 。 
实现 按 条 件 查询 ， 如 按 用 户 指定 条 件 查询 、 按 日 期 时 间 段 查询 、 按 综合 条 件 查询 等 。 
要 求 记录 图 书 借阅 、 续 借 、 归 还 时 进行 操作 的 操作 员 。 
实现 对 图 书 借阅 、 续 借 和 归还 过 程 的 全 程 数据 信息 跟踪 。 
提供 借阅 到 期 提醒 功能 ， 使 管理 者 可 以 及 时 了 解 已 经 到 归还 日 期 的 图 书 借阅 信息 。 
提供 灵活 、 方 便 的 权限 设置 功能 ， 使 整个 系统 的 管理 分 工 明确 。 
具有 易 维护 性 和 易 操 作 性 。 


因 因 因 办 办 办 办 办 办 办 多 


28.3 系统 设计 


28.3.1 系统 目标 


根据 前 面 所 做 的 需求 分 析 及 用 户 的 需求 可 以 得 出 ， 学 校 图 书馆 管理 系统 实施 后 ， 应 达到 以 下 目标 。 
回 ”网 站 设计 页 面 要 求 美观 大 方 、 功 能 全 面 ， 操 作 简单 。 

回 ”网 站 整体 结构 和 操作 流程 合理 顺畅 ， 实 现 人 性 化 设计 。 

规范 、 完 善 的 基础 信息 设置 。 


加 


全 
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PHP+MySQL 开发 实战 


对 操作 员 设 置 不 同 的 操作 权限 ， 为 管理 员 提供 修改 权限 。 
对 所 有 读者 进行 集中 管理 。 

对 图 书信 息 进行 集中 管理 。 

实现 图 书 借阅 排行 ， 以 便 了 解 当 前 的 畅销 书 。 

提供 快速 的 图 书信 息 、 图 书 借阅 检索 功能 。 

实现 图 书 借阅 、 图 书 续 借 、 图 书 归还 功能 。 

实现 按 条 件 查询 ， 如 按 用 户 指定 条 件 查询 、 按 日 期 时 间 段 查询 、 按 综合 条 件 查询 等 。 
实现 记录 图 书 借阅 、 续 借 、 归 还 时 进行 操作 的 操作 员 。 
支持 图 书 到 期 提醒 功能 。 

为 操作 员 提 供 密码 修改 功能 。 

系统 运行 稳定 、 安 全 可 靠 。 


系统 功能 结构 


学 校 图 书馆 管理 系统 的 特点 ， 可 以 将 其 分 为 系统 设置 、 读 者 管理 、 图 书 档案 管理 、 图 书 借 还 和 系 
个 部 分 ， 其 中 各 个 部 分 及 其 包括 的 具体 功能 模块 如 图 28.1 所 示 。 


学 校 图 书馆 管理 系统 
图 书 档案 管理 


证 灾 让 机 天 
闹 吐 码 准 瑟 闻 
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28.1 学 校 图 书馆 管理 系统 功能 结构 图 


系统 流程 图 
图 书馆 管理 系统 的 流程 如 图 28.2 所 示 。 


系统 预览 


图 书馆 管理 系统 由 多 个 程序 页 面 组 成 ， 下 面 仅 列 出 几 个 典型 页 面 ， 其 他 页 面 参 见 光盘 中 的 源 程序 。 
登录 页 面 如 图 28.3 所 示 ， 该 页 面 用 于 实现 管理 员 登 录 。 系 统 首页 如 图 28.4 所 示 ， 该 页 面 用 于 实现 
导航 、 图 书 借阅 排行 和 版 权 信息 等 功能 。 
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图 书 借阅 页 面 如 图 28.5 所 示 ， 该 页 面 用 于 实现 图 书 借阅 功能 。 图 书 借阅 查询 页 面 如 图 28.6 所 示 ， 该 页 
面 用 于 实现 按照 复合 条 件 查询 图 书 借阅 信息 。 


图 28.5 图 书 借阅 图 28.6 ”图书 借阅 查询 
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图 书 借阅 页 面 如 图 28.5 所 示 ， 该 页 面 用 于 实现 图 书 借阅 功能 。 图 书 借阅 查询 页 面 如 图 28.6 所 示 ， 该 页 
面 用 于 实现 按照 复合 条 件 查询 图 书 借阅 信息 。 
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28.3.5 “文件 夹 组 织 结 构 


锋 1 视频 讲解 :光盘 \TM\Video\ 第 28 章 \ 文 件 夹 组 织 结构 .exe 
在 编写 代码 前 , 可 以 先 把 系统 中 可 能 用 到 的 文件 


久 © comn 用 于 存储 网 站 使 用 的 数据 库 连 接 文件 
夹 创建 出 来 (如 创建 一 个 名 为 Images 的 文件 夹 ， 用 ts ln 
于 保存 网 站 中 所 使 用 的 图 片 》， 这样 不 但 可 以 方便 以 i RT 


后 的 开发 工作 ,也 可 以 规范 网 站 的 整体 架构 。 笔 者 在 
开发 学 校 图 书馆 管理 系统 时 ， 设 计 了 如 图 28.7 所 示 图 28.7 文件 夹 组 织 结构 
的 文件 夹 组 织 结构 。 在 开发 时 ， 只 需要 将 所 创建 的 文件 保存 在 相应 的 文件 夹 中 即 可 。 


28.4 ”数据 库 设计 


如 1 视频 讲解 : 光盘 \TMNVideo\ 第 28 章 \ 数 据 库 设计 .exe 
学 校 图 书馆 管理 系统 是 一 个 数据 库 开 发 的 Web 网站。 下 面 对 该 系统 使 用 的 数据 库 进行 分 析 和 介绍 。 


28.4.1 数据 库 分 析 

由 于 本 系统 是 为 中 小 型 图 书馆 开发 的 程序 , 需要 充分 考虑 成 本 及 使 用 需求 (如 跨 平台 ) 等 问题 ,而 MySQL 
是 世界 上 最 为 流行 的 开放 源 代码 的 数据 库 ， 是 完全 网 络 化 的 跨 平台 的 关系 型 数据 库 系统 ， 这 正好 满足 了 中 
小 型 企业 的 需求 ， 所 以 本 系统 采用 MySQL 数据 库 。 


28.4.2 ”数据 库 概念 设计 


根据 对 系统 所 做 的 需求 分 析 、 系 统 设计 ， 规 划 出 本 系统 中 使 用 的 数据 库 实体 分 别 为 图 书 档案 实体 、 读 
者 档案 实体 、 借 阅 档案 实体 、 归 还 档案 实体 和 管理 员 实 体 。 下 面 将 介绍 几 个 关键 实体 的 E-R 图 。 

1. 图 书 档案 实体 

图 书 档案 实体 包括 编号 、 条 形 码 、 书 名 、 类 型 、 作 者 、 译 者 、 出 版 社 、 价 格 、 页 码 、 书 架 、 录 入 时 间 
和 操作 员 等 属性 。 图 书 档案 实体 的 E-R 图 如 图 28.8 所 示 。 


Ce 


28.8 图 书 档案 实体 E-R 图 
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Ce 


28.8 图 书 档案 实体 E-R 图 
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2. 读者 档案 实体 
读者 档案 实体 包括 编号 、 姓 名 、 性 别 、 条 形 码 、 职 业 、 出 生日 期 、 有 效 证 件 、 证 件 号 码 、 电 话 、 电 子 
邮件 、 登 记 日 期 、 操 作 员 、 类 型 和 备注 等 属性 。 读 者 档案 实体 的 E-R 图 如 图 28.9 所 示 。 


电子 邮 


28.9 读者 档案 实体 E-R 图 


3. 借阅 档案 实体 
借阅 档案 实体 包括 编号 、 读 者 编号 、 图 书 编号 、 借 书 时 间 、 应 还 时 间 、 操 作 员 和 是 否 归 还 属性 。 借 阅 
档案 实体 的 E-R 图 如 图 28.10 所 示 。 


4. 归还 档案 实体 
归还 档案 实体 包括 编号 、 读 者 编号 、 图 书 编号 、 归 还 时 间 和 操作 员 属 性 。 归 还 档案 实体 的 E-R 图 如 


图 28.11 所 示 。 
图 书 编号 


归还 


28.10 ”借阅 档案 实体 E-R 28.11 “归还 档案 实体 E-R 


28.4.3 创建 数据 库 及 数据 表 


合 实际 情况 及 对 用 户 需求 的 分 析 ， 学 校 图 书馆 管理 系统 db_library 数据 库 主 要 包含 如 下 10 个 数据 表 ， 
如 图 28.12 所 示 。 
下 面 介绍 几 个 主要 数据 表 的 结构 。 
1. tb_bookinfo (图 书信 息 表 ) 
图 书信 息 表 主 要 用 于 存储 图 书 的 基础 信息 。 该 数据 表 的 结构 如 图 28.13 所 示 。 


四 
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图标 务 器 :lecalhost ”县 效 据 库 : db_library ， 四 表 : tb bookinfo 
子 和 全 美 型 整 蛙 Nul 材 认 星 外 说 明 
Barcode 。 warehar30) ab2313_ehi 否 | 
hookname varchar(TU) ob2312_chir 是 WUE 图 书 名 称 
加 服务 器 : localhest ， 局 数据 库 : db_library pead it 四 是 NU 图 类型 
表 类 型 可 加 党 明 aor varchar(3l) ob2312_chinase_ti 是 NULL 国 书 作 老 
也 _hookcase 。 MISAM go2312_onines9_ci 。 图 肯 和 要 信息 表 Wansiator 。 vartharG0) ob2312_chinasa_el 是 WULC 3 者 
了 tb_hookinfe MHSAM go2311_cninase_ci 。 硬 忆 六 自 表 ES VarharGm) 0b2312_chnesa zl 是 。 NULL SBN 
thhorrow WSAM 1 ice falaD 是 MOLL 人 
th_lbray MSAM page inxID 是 AULL 图 书页 码 
th_manager WSAM a bookcase mlol 是 ML ET 
tb_parameter WSAM 002317_cninase_c| 。 基 涩 设 填 呈 肝素 storage WII 是 NULL 加 节 隆 存 
由 _publlshing 。 M3AM gb2312_cninese_cl 。 出 版 诗 信息 者 mme da 是 UL 入 用 时间 
四 puremw WISAM gn2312_cninase_cl 。 杭 卫 结 县 表 operator varchar(30) gb2312_chinese_ri 是 。 NULL 加 作品 
tb reader MBAM 。 购 2313_chinese_cl 。 证 者 伟 息 才 el tmynttt) 是 NULL 是 百 了 际 
也 _readertype M3AM gh2312_chinese_ci 。 评 考 兴 型 信息 才 a ix 是 NULL ab increment ga 只 
图 28.12 学 校 图 书馆 管理 系统 数据 表 图 28.13 ”图 书信 息 表 结构 
2. tb_borrow (图 书 借阅 信息 表 ) 
图 书 借阅 信息 表 主 要 用 于 存储 图 书 的 借阅 信息 。 该 数据 表 的 结构 如 图 28.14 所 示 。 
3. tb_reader (读者 信息 表 ) 
读者 信息 表 主 要 用 于 存储 读者 的 基础 信息 。 该 数据 表 的 结构 如 图 28.15 所 示 。 
图 服务 器 : lecalhost 居 数据 库 ; db_libray ， 国 末 : 了 b_reader 
+8 Rn 镍 外 说 明 
a mk) 青 auto_neremert 和 友 过 
nome vartha20l 9h2312 shese_el 是 NULL 省 扣 
sox Varthar 。 92512.phinese_d 是 NULL 4 
boreode 。 warthapo 9k2312 shese-ol 是 NULL 计 玫 到 
vocation varchars0) gb2312_chinase_el 是 WULL 读者 黄 业 
加 服务 器 :localhest ， 尿 数据 库 : db_library 、 回 表 : th_borrew RE 侈 本 WR 出 和 期 
学 县 类 型 整理 un 黑人 往外 吏 明 paperfype -varcharit0) 。 屿 2312_shinaae_cl 是 NULL 读 省 区 件 
过 an) 否 uic_jncremant 。 Bb 柜 呈 过 Paper varchari20) gk2312_chinese_cl 是 NULL 证 和 8 码 
rondong mam 是 ML a tal varchari2D) gb2312.ehinese_d 是 AULL 读者 电 话 
bookid Ito 是 NL 图 Hid 护 S ma varchartt00) 2312_chinese_el 是 NULL Email 二 址 
barrowime da 是 WL EE creatoDate cate 是 NULL bk 
backTime 。 da 是 Mt 四 Hg 二 oporstor 。 warthap0 9k2312 shese_el 是 NULL 妆 作 有 
operator varchar(30) gb2312chinese_c 是 NOLL 换 作 局 remark mediumtst 。 gb2312_shinsse_cl 是 NULL 3 
thack yin) 是 0 是 下 peid okt) 是 NULL ET 
图 28.14 图 书 借阅 信息 表 结构 图 28.15 读者 信息 表 结构 


J/ 
说 明 限于 篇 幅 ， 笔 者 在 此 只 给 出 较 重要 的 数据 表 ， 其 他 数据 表 参 见 本 书 附带 的 光盘 。 


28.5 首页 设计 
车 视频 讲解 : 光盘 \TMIVVideo\ 第 28 章 \ 首 页 设计 .exe 


28.5.1 首页 概述 


管理 员 通过 系统 登录 模块 的 验证 后 ， 可 以 登录 到 图 书馆 管理 系统 的 首页 。 系 统 首页 主要 包括 导航 栏 、 
排行 榜 和 版 权 信息 3 部 分 。 其 中 ， 导 航 栏 中 的 功能 菜单 将 根据 登录 管理 员 的 权限 进行 显示 。 例 如 ， 系 统管 
理 员 MR 登录 后 ， 将 拥有 整个 系统 的 全 部 功能 ， 因 为 它 是 超级 管理 员 。 

下 面 看 一 下 本 实例 中 提供 的 系统 首页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\28\index.php， 如 图 28.16 所 示 。 
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thack yin) 是 0 是 下 peid okt) 是 NULL ET 
图 28.14 图 书 借阅 信息 表 结构 图 28.15 读者 信息 表 结构 


J/ 
说 明 限于 篇 幅 ， 笔 者 在 此 只 给 出 较 重要 的 数据 表 ， 其 他 数据 表 参 见 本 书 附带 的 光盘 。 


28.5 首页 设计 
车 视频 讲解 : 光盘 \TMIVVideo\ 第 28 章 \ 首 页 设计 .exe 


28.5.1 首页 概述 


管理 员 通过 系统 登录 模块 的 验证 后 ， 可 以 登录 到 图 书馆 管理 系统 的 首页 。 系 统 首页 主要 包括 导航 栏 、 
排行 榜 和 版 权 信息 3 部 分 。 其 中 ， 导 航 栏 中 的 功能 菜单 将 根据 登录 管理 员 的 权限 进行 显示 。 例 如 ， 系 统管 
理 员 MR 登录 后 ， 将 拥有 整个 系统 的 全 部 功能 ， 因 为 它 是 超级 管理 员 。 

下 面 看 一 下 本 实例 中 提供 的 系统 首页 ， 该 页 面 在 本 书 光盘 中 的 路 径 为 \TM\28\index.php， 如 图 28.16 所 示 。 


局 
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放 者 定价 (元 ) 仙 网 交 堵 


1 155421 。 PE 人才 区 专 大 全 pp Ra 多 . 当 和 加 4 
2 ronlisistio! Visw Basic 件 困 基 大 全 ”计生 机 在 友 ype 人 Rapab 太 入 而 哲 \XHV 66.00 3 
3 lene MR a ati Rehabilt PA e500 2 


129454321 了 开发 实战 2xc 例 宇 虎 系列 。。 BE 书 弄 ”人 民 让 出 版 社 剂 中 华 、 潘 由 第 6.o 1 


Cophitht © 2007 we orbecd cm a 反 权 信息 | 
版 权 信息 
玉 站 请 使 用 IE 6 0 或 以 上 版 本 1024*768 力 至 住 哇 示 小 采 


图 28.16 学校 图 书馆 管理 系统 首页 


28.5.2 ”权限 设置 技术 


学 校 图 书馆 管理 系统 是 一 个 功能 全 面 的 大 型 Web 网 站 ， 出 于 对 网 站 安全 性 的 考虑 ， 本 网 站 对 该 系统 进 
行 权限 的 分 配 ， 只 有 管理 员 级 别 的 超级 用 户 可 以 对 普通 用 户 的 权限 进行 管理 和 设置 。 系 统 首页 主要 通过 判 
断 管理 员 的 权限 来 显示 该 用 户 所 操作 的 功能 模块 。 其 关键 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\InstanceW28\navigation.php) 


<?php 
session_start(); /初始 化 session 变量 
include("conn/conn.php"); // 连 接 数据 库 文件 


$query=mysql_query("select m.id,m.name,p.id,p.sysset,p.readerset,p.bookset,p.borrowback,p.sysquery from 
tb_manager as m left join (select * from tb_purview ) as p on m.id=p.id where name='$_SESSION 
[admin_name]"); 

Sinfo=mysql_fetch_array($query); /检索 用 户 权限 

?> 

<!- 检 索 用 户 所 对 应 的 权限 ， 如 果 权 限 值 为 1， 则 说 明 该 功能 可 用 ， 并 输出 到 浏览 器 ， 和 否则 不 显示 --> 

<td width="170%" align="right"> 

<a href="index.php" class="a1"> 首 页 </a> | 

<?php if($info[syssetj==1){ ?><a _ onmouseover=showmenu(event,sysmenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="a1"> 系 统 设置 </a> | <?php } ?> 

<?php if($info[readerset]j==1){?><a onmouseover=showmenu(event,readermenu) onmouseout=delayhidemenu() 


style="CURSOR:hand" ="a1"> 读 者 管理 </a> | <?php } ?> 
<?php if($info[booksetlj==1){ ?><a href="book.php" class="a1"> 图 书 档案 管理 </a> | <?php }?> 
<?php if($info[borrowback]==1){?><a ‘onmouseover=showmenu(event,borrowmenu) 


‘onmouseout=delayhidemenu() style="CURSOR:hand"class="a1" > 图 书 借 还 </a> | <?php }?> 

<?php if($info[sysquery]==1) ?><a onmouseover=showmenu(event,querymenu) onmouseout=delayhidemenu() 
style="CURSOR:hand" class="a1"> 系 统 查询 </a> | <?php } ?> 

<a href="pwd_Modify.php" class="a1"> 更 改口 令 </a> | 

<a href="safequit.php" class="a1"> 注 销 </a> 

</td> 


of 
于 培 明 在 权限 信息 表 tb_purview 中 ， 权 限 值 为 1， 代表 具备 该 模块 的 操作 权限 ; 权限 值 为 0， 代 表 不 


具备 该 模块 的 操作 权限 。 
_- 国 ) 
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在 实现 系统 导航 菜单 时 ， 引 用 了 JS 文件 menuJS， 该 文件 中 包含 全 部 实现 半 透 明 背 景 菜单 的 JavaScript 
代码 。 


-2 
[7 将 页 面 中 所 涉及 的 JavaScript 代码 保存 在 一 个 单独 的 JS 文件 中 ， 然 后 通过 <script></script> 引 
用 到 需要 的 页 面 ， 可 以 规范 页 面 代码 。 在 系统 导航 页 面 引用 menuLJS 文件 的 代码 如 下 : 


<script src="JS/menu.JS"></script> 


28.5.3 ”首页 的 实现 过 程 


系统 首页 的 内 容 显示 区 用 于 显示 图 书 的 排行 信息 ， 并 将 排行 结果 按 借阅 数量 降序 排列 。 该 页 的 关键 代 

码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\28\index.php) 

<?php 

include("conn/conn.php"); /| 连接 数据 源 文 件 

$sql=mysql_query("select * from (select bookid,count(bookid) as degree from tb_borrow group by bookid) as 

borr join (select b.*,c.name as bookcasename,p.pubname,t.typename from tb_bookinfo b left join tb_bookcase c 

on b.bookcase=c.id join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as 

book on borr.bookid=book.id order by borr.degree desc limit 10"); 


$info=mysql_fetch_array($sql); /检索 图 书 借阅 信 息 

Si=1; 

do{ /应 用 do…while 循环 语句 显示 图 书信 息 
?> 

<tr> 


<td height="25" align="center"><?php echo $i;?></td> 

<td style="padding:5px;">&nbsp;<?php echo S$info[barcode];?></td> 
<td style="padding:5px;"><?php echo S$info[bookname];?></td> 

<td style="padding:5px;"><?php echo S$info[typename];?></td> 

<td align="center">&nbsp;<?php echo $info[bookcasename];?></td> 
<td align="center">&nbsp;<?php echo $info[pubname];?></td> 

<td align="center"><?php echo S$info[author];?></td> 

<td align="center"><?php echo S$info[price];?></td> 

<td align="center"><?php echo S$info[degree];?></td> 


</tr> 
<?php 
S$i=$i+1; // 变 量 自 加 1 操作 
}while($info=mysql_fetch_array($sq])); /ldo…while 循环 语句 结束 
> 


28.6 ”管理 员 模 块 设计 
区 4 视频 讲解 : 光盘 \IM\Video\ 第 28 章 \ 管 理 员 模块 设计 .exe 


28.6.1 管理 员 模 块 概述 


管理 员 模 块 主要 包括 管理 员 登 录 、 查 看 管理 员 列 表 、 添 加 管理 员 信息 、 管 理 员 权限 设置 、 删 除 管理 员 


@_ 


PHP+MySQL 开发 实战 


在 实现 系统 导航 菜单 时 ， 引 用 了 JS 文件 menuJS， 该 文件 中 包含 全 部 实现 半 透 明 背 景 菜单 的 JavaScript 
代码 。 


-2 
[7 将 页 面 中 所 涉及 的 JavaScript 代码 保存 在 一 个 单独 的 JS 文件 中 ， 然 后 通过 <script></script> 引 
用 到 需要 的 页 面 ， 可 以 规范 页 面 代码 。 在 系统 导航 页 面 引用 menuLJS 文件 的 代码 如 下 : 


<script src="JS/menu.JS"></script> 


28.5.3 ”首页 的 实现 过 程 


系统 首页 的 内 容 显示 区 用 于 显示 图 书 的 排行 信息 ， 并 将 排行 结果 按 借阅 数量 降序 排列 。 该 页 的 关键 代 

码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\28\index.php) 

<?php 

include("conn/conn.php"); /| 连接 数据 源 文 件 

$sql=mysql_query("select * from (select bookid,count(bookid) as degree from tb_borrow group by bookid) as 

borr join (select b.*,c.name as bookcasename,p.pubname,t.typename from tb_bookinfo b left join tb_bookcase c 

on b.bookcase=c.id join tb_publishing p on b.ISBN=p.ISBN join tb_booktype t on b.typeid=t.id where b.del=0) as 

book on borr.bookid=book.id order by borr.degree desc limit 10"); 


$info=mysql_fetch_array($sql); /检索 图 书 借阅 信 息 

Si=1; 

do{ /应 用 do…while 循环 语句 显示 图 书信 息 
?> 

<tr> 


<td height="25" align="center"><?php echo $i;?></td> 

<td style="padding:5px;">&nbsp;<?php echo S$info[barcode];?></td> 
<td style="padding:5px;"><?php echo S$info[bookname];?></td> 

<td style="padding:5px;"><?php echo S$info[typename];?></td> 

<td align="center">&nbsp;<?php echo $info[bookcasename];?></td> 
<td align="center">&nbsp;<?php echo $info[pubname];?></td> 

<td align="center"><?php echo S$info[author];?></td> 

<td align="center"><?php echo S$info[price];?></td> 

<td align="center"><?php echo S$info[degree];?></td> 


</tr> 
<?php 
S$i=$i+1; // 变 量 自 加 1 操作 
}while($info=mysql_fetch_array($sq])); /ldo…while 循环 语句 结束 
> 


28.6 ”管理 员 模 块 设计 
区 4 视频 讲解 : 光盘 \IM\Video\ 第 28 章 \ 管 理 员 模块 设计 .exe 


28.6.1 管理 员 模 块 概述 


管理 员 模 块 主要 包括 管理 员 登 录 、 查 看 管理 员 列 表 、 添 加 管理 员 信息 、 管 理 员 权限 设置 、 删 除 管理 员 


@_ 
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和 更 改口 令 6 个 功能 。 管 理 员 模 块 的 框架 图 如 图 28.17 所 示 。 


管理 员 列 表 


rg 


添加 管理 员 权限 设置 删除 管理 员 [| wan [| 


图 28.17 管理 员 模 块 的 框架 图 
28.6.2 ”控制 文件 的 访问 权限 
在 管理 员 模块 中 ， 涉 及 到 的 数据 表 是 tb_manager (管理 员 信息 表 ) 和 tb _purview (权限 表 ) 。 其 中 ， 管 


理 员 信息 表 中 保存 的 是 管理 员 名 称 和 密码 等 信息 ， 权 限 表 中 保存 的 是 各 管理 员 的 权限 信息 ， 这 两 个 表 通过 
各 自 的 这 字段 相关 联 。 通 过 这 两 个 表 可 以 获得 完整 的 管理 员 信息 。 


et 
MR、 密 码 为 mrsoft、 拥 有 所 有 权限 )， 即 在 MySQL 的 客户 端 命令 行 中 应 用 下 面 的 语句 分 别 向 管理 员 信 
息 表 tb_manager 和 权限 表 tb_purview 中 添加 一 条 数据 。 


# 添 加 管理 员 信息 

insert into tb_manager (name,pwd) values('MR"','mrsoft’); 

# 添 加 权限 信息 

insert into tb_purview values(1,1,1,1,1,1); 

从 网 站 安全 的 角度 考虑 ， 仅 仅 有 上 面 介绍 的 系统 登录 页 面 并 不 能 有 效 地 保证 系统 的 安全 ， 一 旦 系统 首 
页 面 的 地 址 被 他 人 获得 ， 就 可 以 通过 在 地 址 栏 中 输入 系统 的 首页 面 地 址 而 直接 进入 到 系统 中 。 为 了 便于 网 
站 的 维护 ， 将 验证 用 户 是 否 登 录 的 代码 封装 在 独立 的 PHP 文件 中 ， 即 check login.php 文件 。 验 证 用 户 是 否 
登录 的 具体 代码 如 下 : 

(代码 位 置 ， 光盘 \TMNInstance\28\check_login.php) 

<?php 

session_start(); 

if(lisset($_SESSION[admin_name'])X{ 

echo "<script>window.location.href="login.php';</script>"; 
计 
?> 


当 系 统 调用 首页 时 ， 会 判断 session 变量 admin name 是 否 存在 ， 如 果 不 存在 ， 将 页 面 重 定 向 到 系统 登 


录 (loginphp) 页 面 。 
@ 
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和 更 改口 令 6 个 功能 。 管 理 员 模 块 的 框架 图 如 图 28.17 所 示 。 


管理 员 列 表 


rg 


添加 管理 员 权限 设置 删除 管理 员 [| wan [| 


图 28.17 管理 员 模 块 的 框架 图 
28.6.2 ”控制 文件 的 访问 权限 
在 管理 员 模块 中 ， 涉 及 到 的 数据 表 是 tb_manager (管理 员 信息 表 ) 和 tb _purview (权限 表 ) 。 其 中 ， 管 


理 员 信息 表 中 保存 的 是 管理 员 名 称 和 密码 等 信息 ， 权 限 表 中 保存 的 是 各 管理 员 的 权限 信息 ， 这 两 个 表 通过 
各 自 的 这 字段 相关 联 。 通 过 这 两 个 表 可 以 获得 完整 的 管理 员 信息 。 


et 
MR、 密 码 为 mrsoft、 拥 有 所 有 权限 )， 即 在 MySQL 的 客户 端 命令 行 中 应 用 下 面 的 语句 分 别 向 管理 员 信 
息 表 tb_manager 和 权限 表 tb_purview 中 添加 一 条 数据 。 


# 添 加 管理 员 信息 

insert into tb_manager (name,pwd) values('MR"','mrsoft’); 

# 添 加 权限 信息 

insert into tb_purview values(1,1,1,1,1,1); 

从 网 站 安全 的 角度 考虑 ， 仅 仅 有 上 面 介绍 的 系统 登录 页 面 并 不 能 有 效 地 保证 系统 的 安全 ， 一 旦 系统 首 
页 面 的 地 址 被 他 人 获得 ， 就 可 以 通过 在 地 址 栏 中 输入 系统 的 首页 面 地 址 而 直接 进入 到 系统 中 。 为 了 便于 网 
站 的 维护 ， 将 验证 用 户 是 否 登 录 的 代码 封装 在 独立 的 PHP 文件 中 ， 即 check login.php 文件 。 验 证 用 户 是 否 
登录 的 具体 代码 如 下 : 

(代码 位 置 ， 光盘 \TMNInstance\28\check_login.php) 

<?php 

session_start(); 

if(lisset($_SESSION[admin_name'])X{ 

echo "<script>window.location.href="login.php';</script>"; 
计 
?> 


当 系 统 调用 首页 时 ， 会 判断 session 变量 admin name 是 否 存在 ， 如 果 不 存在 ， 将 页 面 重 定 向 到 系统 登 


录 (loginphp) 页 面 。 
@ 
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28.6.3 ”系统 登录 页 面 的 实现 过 程 


系统 登录 是 进入 学 校 图 书馆 管理 系统 的 入 口 ， 
主要 用 于 验证 管理 员 的 身份 。 运行 本 系统 ， 首 先进 
入 的 是 系统 登录 页 面 ,在 该 页 面 中 ， 系 统管 理 员 可 
以 通过 输入 正确 的 管理 员 名 称 和 密码 登录 到 系统 
首页 ， 当 用 户 没有 输入 管理 员 名 称 或 密码 时 ， 系 统 
会 通过 JavaScript 进行 判断 ， 并 给 予 信息 提示 。 系 
统 登 录 页 面 的 运行 结果 如 图 28.18 所 示 。 

系统 登录 页 面 主要 用 于 收集 管理 员 的 输入 信 
息 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 
否 为 空 。 该 页 面 所 涉及 到 的 表单 元 素 如 表 28.1 所 示 。 


表 28.1 系统 登录 页 面 所 涉及 的 表单 元 素 


28.18 ”系统 登录 页 面 的 运行 结果 


名 称 洛 -” 叉 
forml 管理 员 登 录 表单 
name 管理 员 名 称 
pwd 管理 员 密 码 
submit “确定 ”按钮 
submit3 “ 重 置 ” 按 钮 
submit2 “关闭 ”按钮 
编写 自 定义 的 JavaScript 函数 ， 用 于 判断 管理 员 名 称 和 密码 是 否 为 空 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TMNInstance\28\check_login.php) 
<script language="javascript"> 
function check(form){ // 自 定义 一 个 JavaScript 函数 check() 
if (form.name.value=="™")}{ // 如 果 管理 员 名 称 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 管理 员 名 称 ");form.name .focus();return false; 
} 
if (form.pwd.value==" // 如 果 管理 员 密 码 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 密码 !");form.pwd .focus();return false; 
} 
} 
</script> 


提交 表单 到 数据 处 理 页 ， 页 面 中 为 了 防止 非法 用 户 进入 学 校 图 书馆 管理 系统 首页 ， 通 过 调用 chkinputO 
方法 判断 用 户 名 和 密码 是 否 正确 。 如 果 为 合法 用 户 ， 则 可 以 登录 学 校 图 书馆 管理 系统 的 首页 ， 否 则 ， 弹 出 
相应 的 错误 提示 。 其 关键 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\28\chklogin.php) 


<?php 
session_start(); /初始 化 session 变量 
$A_name=$_POSTIname]; /接收 表单 提交 的 用 户 名 
$A_pwd=$_POST[Ipwd]; /接收 表单 提交 的 密码 
@ casschkinput{ /定义 类 

var $name; 

Var $pwd; 


局 
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28.6.3 ”系统 登录 页 面 的 实现 过 程 


系统 登录 是 进入 学 校 图 书馆 管理 系统 的 入 口 ， 
主要 用 于 验证 管理 员 的 身份 。 运行 本 系统 ， 首 先进 
入 的 是 系统 登录 页 面 ,在 该 页 面 中 ， 系 统管 理 员 可 
以 通过 输入 正确 的 管理 员 名 称 和 密码 登录 到 系统 
首页 ， 当 用 户 没有 输入 管理 员 名 称 或 密码 时 ， 系 统 
会 通过 JavaScript 进行 判断 ， 并 给 予 信息 提示 。 系 
统 登 录 页 面 的 运行 结果 如 图 28.18 所 示 。 

系统 登录 页 面 主要 用 于 收集 管理 员 的 输入 信 
息 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 
否 为 空 。 该 页 面 所 涉及 到 的 表单 元 素 如 表 28.1 所 示 。 


表 28.1 系统 登录 页 面 所 涉及 的 表单 元 素 


28.18 ”系统 登录 页 面 的 运行 结果 


名 称 洛 -” 叉 
forml 管理 员 登 录 表单 
name 管理 员 名 称 
pwd 管理 员 密 码 
submit “确定 ”按钮 
submit3 “ 重 置 ” 按 钮 
submit2 “关闭 ”按钮 
编写 自 定义 的 JavaScript 函数 ， 用 于 判断 管理 员 名 称 和 密码 是 否 为 空 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TMNInstance\28\check_login.php) 
<script language="javascript"> 
function check(form){ // 自 定义 一 个 JavaScript 函数 check() 
if (form.name.value=="™")}{ // 如 果 管理 员 名 称 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 管理 员 名 称 ");form.name .focus();return false; 
} 
if (form.pwd.value==" // 如 果 管理 员 密 码 为 空 ， 则 弹出 提示 信息 ， 并 重新 返回 焦点 
alert(" 请 输入 密码 !");form.pwd .focus();return false; 
} 
} 
</script> 


提交 表单 到 数据 处 理 页 ， 页 面 中 为 了 防止 非法 用 户 进入 学 校 图 书馆 管理 系统 首页 ， 通 过 调用 chkinputO 
方法 判断 用 户 名 和 密码 是 否 正确 。 如 果 为 合法 用 户 ， 则 可 以 登录 学 校 图 书馆 管理 系统 的 首页 ， 否 则 ， 弹 出 
相应 的 错误 提示 。 其 关键 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\28\chklogin.php) 


<?php 
session_start(); /初始 化 session 变量 
$A_name=$_POSTIname]; /接收 表单 提交 的 用 户 名 
$A_pwd=$_POST[Ipwd]; /接收 表单 提交 的 密码 
@ casschkinput{ /定义 类 

var $name; 

Var $pwd; 


局 
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function chkinput($x,$y){ /定义 一 个 方法 
Sthis->name=$x; // 将 管理 员 名 称 传 给 类 对 象 $this->name 
Sthis->pwd=$y; // 将 管理 员 密 码 传 给 类 对 象 $this->pwd 
| 
function checkinput(){ 
include("conn/conn.php"); /连接 数据 库 文件 
$sql=mysql_query("select * from tb_manager where name=".$this->name." and pwd=".$this->pwd."",$conn); 
Sinfo=mysql_fetch_array($sql); // 检 索 管理 员 名 称 和 密码 是 否 正确 
if($info==falseX{ // 如 果 管 理 员 名 称 或 密码 不 正确 , 则 弹出 相关 的 提示 信息 
echo "<script language='javascript>alert(' 您 输入 的 管理 员 名 称 错误 ， 请 重新 输入 ! ');history.back(); 
</script>"; 
exit; 
上 
else{ // 如 果 管理 员 名 称 或 密码 正确 ， 则 弹出 的 相关 提示 信息 


echo "<script>alert(' 管 理 员 登录 成 功 !");window.location='index.php';</script>" 
$_SESSION[admin_name]=Sinfo[name]; // 将 管理 员 名 称 存 到 $_SESSION[admin_name] 变 量 中 
$_SESSION[pwd]=$info[pwd]; // 将 管理 员 密码 存 到 $_SESSION[pwd] 变 量 中 


$obj=new chkinput(trim($name),trim($pwd)); he 
S$obj->checkinput(); 用 类 


yee-- 


wma 
后 用 花 括号 将 类 体 进行 封装 。 
@ new: PHP 中 应 用 new 关键 字 创 建 对 象 。 该 模块 创建 了 一 个 名 为 chkinput 的 验证 管理 员 类 。 
日 $obj->checkinput();: 通过 $obj 对 象 调用 checkinput 类 中 的 属性 和 方法 ， 即 管理 员 名 称 和 密码 。 


28.6.4 ”查看 管理 员 列 表 页 面 的 实现 过 程 


管理 员 登录 后 ， 选 择 “系统 设置 ”/“ 管 理 员 设置 ”命令 ， 进 入 到 查看 管理 员 列表 页 面 。 在 该 页 面 中 ， 
将 以 表格 的 形式 显示 全 部 管理 员 及 其 权限 信息 ， 并 提供 添加 管理 员 信息 、 删 除 管理 员 信息 和 设置 管理 员 权 
限 的 超 链接 。 查 看 管理 员 列表 页 面 的 运行 结果 如 图 28.19 所 示 。 


图 28.19 查看 管理 员 列 表 页 面 的 运行 结果 
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function chkinput($x,$y){ /定义 一 个 方法 
Sthis->name=$x; // 将 管理 员 名 称 传 给 类 对 象 $this->name 
Sthis->pwd=$y; // 将 管理 员 密 码 传 给 类 对 象 $this->pwd 
| 
function checkinput(){ 
include("conn/conn.php"); /连接 数据 库 文件 
$sql=mysql_query("select * from tb_manager where name=".$this->name." and pwd=".$this->pwd."",$conn); 
Sinfo=mysql_fetch_array($sql); // 检 索 管理 员 名 称 和 密码 是 否 正确 
if($info==falseX{ // 如 果 管 理 员 名 称 或 密码 不 正确 , 则 弹出 相关 的 提示 信息 
echo "<script language='javascript>alert(' 您 输入 的 管理 员 名 称 错误 ， 请 重新 输入 ! ');history.back(); 
</script>"; 
exit; 
上 
else{ // 如 果 管理 员 名 称 或 密码 正确 ， 则 弹出 的 相关 提示 信息 


echo "<script>alert(' 管 理 员 登录 成 功 !");window.location='index.php';</script>" 
$_SESSION[admin_name]=Sinfo[name]; // 将 管理 员 名 称 存 到 $_SESSION[admin_name] 变 量 中 
$_SESSION[pwd]=$info[pwd]; // 将 管理 员 密码 存 到 $_SESSION[pwd] 变 量 中 


$obj=new chkinput(trim($name),trim($pwd)); he 
S$obj->checkinput(); 用 类 


yee-- 


wma 
后 用 花 括号 将 类 体 进行 封装 。 
@ new: PHP 中 应 用 new 关键 字 创 建 对 象 。 该 模块 创建 了 一 个 名 为 chkinput 的 验证 管理 员 类 。 
日 $obj->checkinput();: 通过 $obj 对 象 调用 checkinput 类 中 的 属性 和 方法 ， 即 管理 员 名 称 和 密码 。 


28.6.4 ”查看 管理 员 列 表 页 面 的 实现 过 程 


管理 员 登录 后 ， 选 择 “系统 设置 ”/“ 管 理 员 设置 ”命令 ， 进 入 到 查看 管理 员 列表 页 面 。 在 该 页 面 中 ， 
将 以 表格 的 形式 显示 全 部 管理 员 及 其 权限 信息 ， 并 提供 添加 管理 员 信息 、 删 除 管理 员 信息 和 设置 管理 员 权 
限 的 超 链接 。 查 看 管理 员 列表 页 面 的 运行 结果 如 图 28.19 所 示 。 


图 28.19 查看 管理 员 列 表 页 面 的 运行 结果 
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首先 使 用 左 外 联接 语句 (left join…on ) 从 数据 表 tb_manager 和 tb_purview 中 查询 出 符合 条 件 的 数据 ， 
然后 将 查询 结果 应 用 do…while 循环 语句 输出 到 浏览 器 。 其 关键 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance28\manager.php) 
<?php 
include("conn/conn.php"); /连接 数据 库 文件 
© $sql=mysql_query("select m.id,m.name,p.sysset,p.readerset,p.bookset,p.borrowback,p.sysquery from 
tb_manager as m left join (select * from tb_purview) as p on m.id=p.id"); 


$info=mysql_fetch_array($sql); /检索 数据 信息 

dof /应 用 do…while 循环 语句 输出 查询 结果 
?> 

出 符合 查询 条 件 的 记录 人 人 人 on 

<tr> 


<td style="padding:5px;"><?php echo $info[name];?></td> 
四 <tdalign="center"><input name="checkbox" type="checkbox" class="noborder value="checkbox" 
disabled="disabled" <?php if($info[sysset]==1){echo ("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" 

disabled="disabled" <?php if($info[readerset]==1){echo("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info[bookset]==1)}{echo("checked");}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info[borrowback]==1)}{echo("checked"):}?>></td> 

<td align="center"><input name="checkbox" type="checkbox" class="noborder" value="checkbox" disabled 
<?php if($info[sysquery]==1)}{echo("checked");}?>></td> 

<td align="center"><a href="#" onClick="window.open('manager_modify.php?id=<?php echo 
$infofid]; ?>",",'width=292,height=1175')"> 权 限 设置 </a></td> 

<td align="center"><a href="manager_del.php?id=<?php echo $info[id];?>"> 删 除 </a></td> 


</tr> 
/人 
<?php 

}while($info=mysql_fetch_array($sql)); /do…while 循环 语句 结束 
?> 


a 
B 
um 明 @ left join…on: 应 用 左 外 联接 将 两 个 表 或 多 个 表 连 接 起 来 ， 返 回 部 分 或 全 部 匹配 行 ， 详 解 参 
见 13.4.2 节 。 
@ <?php if($info[sysset] 一 1){echo ("checked");}?>: 如 果 系统 设置 字段 的 值 为 1， 则 复 选 框 处 于 选中 


28.6.5 添加 管理 员 信息 页 面 的 实现 过 程 


管理 员 登 录 后 ， 选 择 “系统 设置 ”/“ 管 理 员 设置 ”命令 ， 进 入 
到 查看 管理 员 列 表 页 面 ， 在 该 页 面 中 单 击 “ 添 加 管理 员 信息 ” 超 链 
接 ， 打 开 添 加 管理 员 信息 页 面 。 添 加 管理 员 信息 页 面 的 运行 结果 如 
图 28.20 所 示 。 


人 注意 新 添加 的 管理 员 信息 没有 权限 ， 必 须 通过 设置 管理 员 图 28.20 添加 管理 员 页 面 的 运行 结果 
权限 为 其 指定 可 操作 的 功能 模块 。 


@ 


法 。 
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在 查看 管理 员 列 表 页 面 ， 单 击 “添加 管理 员 信息 ” 超 链 接 的 HIML 代码 如 下 : 


(代码 位 置 ， 光盘 \TM\Instance\28\manager.php) 
<a href="#" onClick="window.open('manager_add.php',","width=292,height=1175')"> 添 加 管理 员 信 息 </a> 


添加 管理 员 页 面 主要 用 于 收集 输入 的 管理 员 信息 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 否 合 
该 页 面 所 涉及 到 的 表单 元 素 如 表 28.2 所 示 。 


表 28.2 添加 管理 员 页 面 所 涉及 的 表单 元 素 


text 


password 


password 


在 添加 管理 员 页 面 中 ， 输 入 合法 的 管理 员 名 称 及 密码 后 ， 单 击 “ 保 存 ” 按 钮 ， 提 交 ee 
理 页 ， 将 添加 的 管理 员 信息 保存 到 数据 表 中 。 如 果 添加 成 功 ， 弹 出 成 功 的 提示 信息 ; 否则， 弹出 错误 提 


其 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\28\manager_ok.php) 
<?php 
include("conn/conn.php"); // 连 接 数据 库 文 件 
if($_POST[submit]!=™"X{ // 如 果 单 击 了 “保存 ”按钮 ， 则 执行 下 面 的 操作 
$name=$_POSTIname]; // 获 取 管理 员 名 称 
S$pwd=$_POST[pwd]; // 获 取 管理 员 密 码 
$sql=mysql_query("insert into tb_manager (name,pwd) values('$name','$pwd')"); 
if($sql==true}{ // 向 数据 表 中 添加 管理 员 信 息 成 功 ， 则 给 出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 添 加 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; 
} 
else{ // 向 数据 表 中 添加 管理 员 信 息 失 败 ， 则 给 出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 添加 失败 ! ');window.close();window.opener.location.reload(); 
</script>"; 
册 
} 
?> 


万 在 添加 管理 员 处理 页 中 应 用 “window.openerlocation reload0:” 语 句 刷新 父 窗口 中 的 信息 。 


28.6.6 ”设置 管理 员 权 限 页 面 的 实现 过 程 
看 htipy/ocalhost32/TM/17/.， le 到 
在 查看 管理 员 列 表 页 面 中 单 击 指定 管理 员 后 面 的 “权限 设置 ” 超 全: 区 
链接 ， 即 可 进入 到 权限 设置 页 面 ， 设 置 该 管理 员 的 操作 权限 。 权 限 设 。 | son, 5 zs 2 ees 
置 页 面 的 运行 结果 如 图 28.21 所 示 。 De 
保存 关闭 


权限 设置 页 面 所 涉及 到 的 表单 元 素 如 表 28.3 所 示 。 


28.21 权限 设置 页 面 的 运行 结果 
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表 28.3 ”权限 设置 页 面 所 涉及 的 表单 元 素 


名 称 元 素 类 型 重要 属性 含 义 
forml form method="post" action="manager modifyokphp” 表单 

id hidden value="<?php echo $info[id]:?>" 管理 员 编 号 
name text value="<?php echo $info[name]:?>" 管理 员 名 称 
Sysset checkbox, <?php if($info[sysset]==1){ echo("checked"):}?> 系统 设置 


Teaderset checkbox <?php if($info[readerset]—1){ echo("checked"):}?> 读者 管理 
bookset checkbox <?php if($info[bookset]==1) {echo("checked"):}?> 图 书 管理 
borrowback checkbox <?php if($info[borrowback]=—1){echo("checked"):}?> 图 书 借 还 
Sysqu checkbox <?php if($info[sysquery}==) {echo("checked"):}?> 系统 查询 


submit submit class="btn grey"” “保存 ”按钮 


Submit2 button Value=" 关 闭 " onClick="window.close0:" “关闭 ”按钮 


在 查看 管理 员 列 表 页 面 中 ， 添 加 “权限 设置 ” 列 ， 并 在 该 列 中 添加 用 于 打开 “权限 设置 ”页 面 的 超 链 
接 代码 ， 代 码 如 下 
〈 代 码 位 置 ， 光 盘 \TMNVInstance\28\manager.php) 
<a href="#" onClick="window .open(manager_modify php?id=<?php echo S$infolid]: ?>',",Wwidth=292, height=1175')'> 
权限 设置 </a> 
从 上 面 的 URL 地 址 中 可 以 获取 设置 管理 员 权限 页 所 涉及 到 的 id 号 ， 将 id 号 提交 给 处 理 页 manager_ 
modifyok.php， 修 改 id 号 所 对 应 的 管理 员 信息 。 其 具体 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\28\manager_modifyok.php) 


<?php 

include("conn/conn.php"); // 连 接 数据 库 文件 

if$_POST[submitll=") // 如 果 提 交 表 单 ， 则 执行 以 下 操作 
$id=$_POST[id]; // 获 取 id 信息 

@ $sysset=$_POST[sysset]==""?0:1; /应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 


Sreaderset=$_POST[readerset]==""?0:1; // 应 用 三 目 运 算 符 求 出 “读者 管理 ” 复 选 框 的 值 


$bookset=$_POST[bookset]==""?0: // 应 用 三 目 运 算 符 求 出 “图 书 管理 ” 复 选 框 的 值 
S$borrowback=$_POST[borrowback]==""?0:1; /应 用 三 目 运算 符 求 出 “图 书 借 还 ” 复 选 框 的 值 
S$sysquery=$_POST[sysquery]==""?0:1; // 应 用 三 目 运 算 符 求 出 “系统 查询 ” 复 选 框 的 值 
$query=mysql_query("select * from tb_purview where id=$id"); 

$info=mysql_fetch_array($query); /检索 权限 信息 表 中 是 否 存在 该 管理 员 
if($info==falseX{ // 如 果 不 存 在 ， 向 权限 表 中 添加 管理 员 权限 信息 


四 mysql_query("insert into tb_purview(id,sysset,readerset,bookset,borrowback,sysquery) 
values($id,$sysset,$readerset,$bookset, $borrowback, $sysquery)"); 


上 
else{ // 否 则 ， 更 新 管理 员 的 权限 信息 
mysql_query("update tb_purview set 
Sysset=$sysset,readerset=$readerset,bookset=$bookset,borrowback=$borrowback,sysquery=$sysquery 


where id="$id"); 

» 

echo"<script language=javascript>alert(' 权 限 设 置 修改 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; // 更 新 成 功 ， 弹 出 提示 信息 ， 并 更 新 父 窗口 

. 

?> 


@ 
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表 28.3 ”权限 设置 页 面 所 涉及 的 表单 元 素 


名 称 元 素 类 型 重要 属性 含 义 
forml form method="post" action="manager modifyokphp” 表单 

id hidden value="<?php echo $info[id]:?>" 管理 员 编 号 
name text value="<?php echo $info[name]:?>" 管理 员 名 称 
Sysset checkbox, <?php if($info[sysset]==1){ echo("checked"):}?> 系统 设置 


Teaderset checkbox <?php if($info[readerset]—1){ echo("checked"):}?> 读者 管理 
bookset checkbox <?php if($info[bookset]==1) {echo("checked"):}?> 图 书 管理 
borrowback checkbox <?php if($info[borrowback]=—1){echo("checked"):}?> 图 书 借 还 
Sysqu checkbox <?php if($info[sysquery}==) {echo("checked"):}?> 系统 查询 


submit submit class="btn grey"” “保存 ”按钮 


Submit2 button Value=" 关 闭 " onClick="window.close0:" “关闭 ”按钮 


在 查看 管理 员 列 表 页 面 中 ， 添 加 “权限 设置 ” 列 ， 并 在 该 列 中 添加 用 于 打开 “权限 设置 ”页 面 的 超 链 
接 代码 ， 代 码 如 下 
〈 代 码 位 置 ， 光 盘 \TMNVInstance\28\manager.php) 
<a href="#" onClick="window .open(manager_modify php?id=<?php echo S$infolid]: ?>',",Wwidth=292, height=1175')'> 
权限 设置 </a> 
从 上 面 的 URL 地 址 中 可 以 获取 设置 管理 员 权限 页 所 涉及 到 的 id 号 ， 将 id 号 提交 给 处 理 页 manager_ 
modifyok.php， 修 改 id 号 所 对 应 的 管理 员 信息 。 其 具体 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\28\manager_modifyok.php) 


<?php 

include("conn/conn.php"); // 连 接 数据 库 文件 

if$_POST[submitll=") // 如 果 提 交 表 单 ， 则 执行 以 下 操作 
$id=$_POST[id]; // 获 取 id 信息 

@ $sysset=$_POST[sysset]==""?0:1; /应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 


Sreaderset=$_POST[readerset]==""?0:1; // 应 用 三 目 运 算 符 求 出 “读者 管理 ” 复 选 框 的 值 


$bookset=$_POST[bookset]==""?0: // 应 用 三 目 运 算 符 求 出 “图 书 管理 ” 复 选 框 的 值 
S$borrowback=$_POST[borrowback]==""?0:1; /应 用 三 目 运算 符 求 出 “图 书 借 还 ” 复 选 框 的 值 
S$sysquery=$_POST[sysquery]==""?0:1; // 应 用 三 目 运 算 符 求 出 “系统 查询 ” 复 选 框 的 值 
$query=mysql_query("select * from tb_purview where id=$id"); 

$info=mysql_fetch_array($query); /检索 权限 信息 表 中 是 否 存在 该 管理 员 
if($info==falseX{ // 如 果 不 存 在 ， 向 权限 表 中 添加 管理 员 权限 信息 


四 mysql_query("insert into tb_purview(id,sysset,readerset,bookset,borrowback,sysquery) 
values($id,$sysset,$readerset,$bookset, $borrowback, $sysquery)"); 


上 
else{ // 否 则 ， 更 新 管理 员 的 权限 信息 
mysql_query("update tb_purview set 
Sysset=$sysset,readerset=$readerset,bookset=$bookset,borrowback=$borrowback,sysquery=$sysquery 


where id="$id"); 

» 

echo"<script language=javascript>alert(' 权 限 设 置 修改 成 功 ! ');window.close();window.opener.location.reload(); 
</script>"; // 更 新 成 功 ， 弹 出 提示 信息 ， 并 更 新 父 窗口 

. 

?> 


@ 
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gm 应 用 三 目 运算 符 求 出 “系统 设置 ” 复 选 框 的 值 ， 如 果 等 于 空 ， 值 
为 0， 和 否则 值 为 1。 
@ insert into: 向 权限 信息 表 中 添加 一 行 数据 信息 ，insert into 语句 只 适用 于 对 单行 数据 的 插入 。 


28.6.7 ”删除 管理 员 的 实现 过 程 


在 查看 管理 员 列 表 页 面 中 ， 单 击 指定 管理 员 信息 后 面 的 “删除 ” 超 链 接 ， 该 管理 员 及 其 权限 信息 将 被 
删除 。 
在 查看 管理 员 列表 页 面 中 添加 以 下 用 于 删除 管理 员 信息 的 超 链接 代码 。 
(代码 位 置 ， 光盘 \TM\Instance\28\manager.php) 
<a href="manager_del.php?id=<?php echo $infofid];?>"> 删 除 </a> 
从 上 面 的 URL 地 址 中 , 可 以 获取 删除 管理 员 所 涉及 到 的 id 号 , 将 id 号 提交 给 manager_del.php 处 理 页 ， 
删除 id 号 所 对 应 的 管理 员 信息 。 其 具体 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\28\manager_del.php) 


<?php 

include("conn/conn.php"); /| 连接 数据 库 文件 

$id=$_GETI[id]; // 获 取 管理 员 的 id 号 

S$sql=mysql_query("delete from tb_manager where id='$id"); /删除 管理 员 表 中 id 号 所 对 应 的 管理 员 信息 

$query=mysql_query("delete from tb_purview where id='$id"); /删除 权限 表 中 id 号 所 对 应 的 管理 员 权限 

if($sql==true and $query==true X{ // 如 果 删 除 操作 成 功 ， 则 弹出 提示 信息 
echo "<script language=javascript>alert(' 管 理 员 删 除 成 功 ! ');history.back();</script>"; 

} 

else{ // 如 果 删 除 操作 失败 ， 人 和 


echo "<script language=javascript>alert(' 管 理 员 删 除 失 败 ! '");history.back();</script>: 


?> 


28.7 图 书 档案 管理 模块 设计 


镶 视频 讲解 :光盘 \TM\Video\ 第 28 章 \ 图 书 档案 管理 模块 设计 .exe 


28.7.1 图 书 档案 管理 模块 概述 


图 书 档案 管理 模块 主要 包括 查看 图 书 列表 、 添 加 图 书信 息 、 修 改 图 书信 息 、 删 除 图 书信 息 和 查看 图 书 
详细 信息 5 个 功能 。 图 书 档案 模块 的 框架 如 图 28.22 所 示 。 


图 书信 息 列表 


y J } y 1 


查看 图 书 列表 | | | | 添加 图 书信 息 | | | | 修改 图 书信 息 [lumseel | | 查看 图 书 详细 信息 


图 28.22 图 书 档案 模块 的 框架 图 
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28.7.2 ”图 书 档案 管理 中 的 多 表 查 询 技术 


在 图 书 档案 管理 模块 中 ， 涉 及 到 的 数据 表 是 tb_bookinfo (图 书信 息 表 ) 、tb_bookcase (书架 设置 表 ) 、 
tb_booktype (图 书 类 型 表 ) 和 tb_publishing (出 版 社 信息 表 ) ， 这 4 个 数据 表 间 通过 相应 的 字段 进行 关联 ， 
如 图 28.23 所 示 ， 通 过 以 上 4 个 表 可 以 获得 完整 的 图 书 档案 信息 。 


th booktype 
ia integer unsigned <pk> tb_bookinfo 
typenane varchar (30) id integer (11) pk> 
days integer unsigned barcode int 
bookname varchar (70) 
Rbooktype——itypeid integer unsigned Ca> 
author var char (30. 
th publishine FR pub | translator varchar (30) 
re 0) le TsB varchar (20) ge> 
pubnane varchar (30) rie ede 
ae integer unsigned 
bookcase integer unsigned <fkl> 
Ebookease [inTine date 
th bookease operator 。 warchar (30) 
I nd GD lm tinyint 1) 
name varchar (30 


图 28.23 ”图书 档案 管理 模块 各 表 间 关系 图 
28.7.3 ”查看 图 书信 息 列表 页 面 的 实现 过 程 
管理 员 登 录 后 ， 选 择 “ 图 书 管理 ”/“ 图 书 档案 管理 ”命令 ， 进 入 到 查看 图 书 列表 页 面 ， 在 该 页 面 中 将 


显示 全 部 图 书信 息 列表 ， 同 时 提供 添加 图 书信 息 、 删 除 图 书信 息 、 修 改 图 书信 息 的 超 链 接 。 查 看 图 书信 息 
列表 页 面 的 运行 结果 如 图 28.24 所 示 。 


(Co 图 书 户 管理 系统 


当 商 位 置 : 图 书 管 承 》 图 和 档案 党 埋 


条 现 


129456789 才 撕 慰 区 人 民 条 电 出 啤 社 。 Ho 书架 [2 
129454921 宝 由 系列 人 Rhu 疡 社 。 ng 书架 全: 员 人 
cresta21 信息 科学 技术 人 民 地 电 册 新 社 。 no 书 加 [ 
aiit5l54I01 计算 机 古训 设计 。。 人 民 亲 电 出 新 针 。 VBH 架 [5 


图 28.24 查看 图 书信 息 列表 页 面 的 运行 结果 


打开 功能 导航 文件 navigation.php， 设 置 “ 图 书 档案 管理 ” 超 链 接 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\InstanceW28\navigation.php) 
<a href="book.php" class="a1"> 图 书 档案 管理 </a> 
首先 应 用 join…on 内 联接 语句 将 tb_bookinfo、tb_bookcase、tb_booktype 和 tb_publishing 4 个 数据 表 连 
接 起 来 ， 检 索 指定 条 件 的 图 书信 息 ， 然 后 应 用 do…while 循环 语句 输出 查询 结果 到 浏览 器 。 查 看 图 书信 息 页 
面 的 代码 如 下 : 


〈 代 码 位 置 : 光盘 \TM\Instance\28\book.php) 
<?php 
include("conn/conn.php"); // 连 接 数据 库 文件 


@ 
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28.7.2 ”图 书 档案 管理 中 的 多 表 查 询 技术 


在 图 书 档案 管理 模块 中 ， 涉 及 到 的 数据 表 是 tb_bookinfo (图 书信 息 表 ) 、tb_bookcase (书架 设置 表 ) 、 
tb_booktype (图 书 类 型 表 ) 和 tb_publishing (出 版 社 信息 表 ) ， 这 4 个 数据 表 间 通过 相应 的 字段 进行 关联 ， 
如 图 28.23 所 示 ， 通 过 以 上 4 个 表 可 以 获得 完整 的 图 书 档案 信息 。 


th booktype 
ia integer unsigned <pk> tb_bookinfo 
typenane varchar (30) id integer (11) pk> 
days integer unsigned barcode int 
bookname varchar (70) 
Rbooktype——itypeid integer unsigned Ca> 
author var char (30. 
th publishine FR pub | translator varchar (30) 
re 0) le TsB varchar (20) ge> 
pubnane varchar (30) rie ede 
ae integer unsigned 
bookcase integer unsigned <fkl> 
Ebookease [inTine date 
th bookease operator 。 warchar (30) 
I nd GD lm tinyint 1) 
name varchar (30 


图 28.23 ”图书 档案 管理 模块 各 表 间 关系 图 
28.7.3 ”查看 图 书信 息 列表 页 面 的 实现 过 程 
管理 员 登 录 后 ， 选 择 “ 图 书 管理 ”/“ 图 书 档案 管理 ”命令 ， 进 入 到 查看 图 书 列表 页 面 ， 在 该 页 面 中 将 


显示 全 部 图 书信 息 列表 ， 同 时 提供 添加 图 书信 息 、 删 除 图 书信 息 、 修 改 图 书信 息 的 超 链 接 。 查 看 图 书信 息 
列表 页 面 的 运行 结果 如 图 28.24 所 示 。 


(Co 图 书 户 管理 系统 


当 商 位 置 : 图 书 管 承 》 图 和 档案 党 埋 


条 现 


129456789 才 撕 慰 区 人 民 条 电 出 啤 社 。 Ho 书架 [2 
129454921 宝 由 系列 人 Rhu 疡 社 。 ng 书架 全: 员 人 
cresta21 信息 科学 技术 人 民 地 电 册 新 社 。 no 书 加 [ 
aiit5l54I01 计算 机 古训 设计 。。 人 民 亲 电 出 新 针 。 VBH 架 [5 


图 28.24 查看 图 书信 息 列表 页 面 的 运行 结果 


打开 功能 导航 文件 navigation.php， 设 置 “ 图 书 档案 管理 ” 超 链 接 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\InstanceW28\navigation.php) 
<a href="book.php" class="a1"> 图 书 档案 管理 </a> 
首先 应 用 join…on 内 联接 语句 将 tb_bookinfo、tb_bookcase、tb_booktype 和 tb_publishing 4 个 数据 表 连 
接 起 来 ， 检 索 指定 条 件 的 图 书信 息 ， 然 后 应 用 do…while 循环 语句 输出 查询 结果 到 浏览 器 。 查 看 图 书信 息 页 
面 的 代码 如 下 : 


〈 代 码 位 置 : 光盘 \TM\Instance\28\book.php) 
<?php 
include("conn/conn.php"); // 连 接 数据 库 文件 


@ 
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$query=mysql_query("select book.barcode,book.id as bookid,book.bookname,bttypename,pb.pubname,bc.name 
from tb_bookinfo book join tb_booktype bt on book.typeid=bt.id join tb_publishing pb on book.ISBN=pb.ISBN 
join tb_bookcase bc on book.bookcase=bc.id"); 


Sresult=mysql_fetch_array($query); // 应 用 内 联接 检索 图 书信 息 
?> 
// 省 略图 书信 息 标题 HTML 标记 部 分 
<?php 
dof /应 用 do…while 循环 语句 输出 查询 结果 
> 
<tr> 


<td style="padding:5px;">&nbsp;<?php echo $result[barcode];?></td> 
<td style="padding:5px;"><a href="book_look.php?id=<?php echo $result[bookid];?>"><?php echo 
$result[bookname];?></a></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[typename];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[pubname];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $resultIname];?></td> 
<td align="center"><a href="book_Modify.php?id=<?php echo $result[bookid];?>"> 修 改 </a></td> 
<td align="center"><a href="book_del.php?id=<?php echo $result[lbookid];?>"> 删 除 </a></td> 
</tr> 


<? 
}while($result=mysql_fetch_array($query)); /ldo…while 循环 语句 结束 
ee 


28.7.4 添加 图 书信 息 页 面 的 实现 过 程 


管理 员 登 录 系 统 后 ， 在 导航 栏 中 单 击 “ 图 书 档案 管理 ” 超 链接 ， 进 入 到 查看 图 书 列表 页 面 。 在 该 页 面 
中 单 击 “添加 图 书信 息 ” 超 链接 ,进入 到 添加 图 书信 息 页 面 。 添 加 图 书信 息 页 面 的 运行 结果 如 图 28.25 所 示 。 


页 。 条 纺 和 吐 读者 管理 。 图 疏导 家 管理 。 医书 漠 寺 。 条 坟 下 而 。 更 孙 口 念 。 注 稍 


作 省 Ca 
译 首 
出 烽 社 RE 
全 检 : 22 E33 
页 码 : Bw EE 
书 可 [r= 
大 |] 返 


28.25 ”添加 图 书信 息 页 面 的 运行 结果 


在 查看 图 书信 息 列表 页 面 中 ， 设 置 “添加 图 书信 息 ” 超 链接 的 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\28\book.php) 
<a href="book_add.php"> 添 加 图 书信 息 </a> 
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添加 图 书信 息 页 面 主要 用 于 收集 输入 的 图 书信 息 以 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 否 合 
法 。 该 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 28.4 所 示 。 
表 28.4 添加 图 书信 息 页 面 所 涉及 的 重要 表单 元 素 


重要 属性 


method="post" action= "book okphp” 


typeld 


isbn 


bookcaseid 


operator 


<?php include("Conn/conn.php"); 
$sql=mysql_query("select * from tb_booktype"); 
Sinfo=mysql fetch array($sq)); 
dof 
?> 
<option value="<?php echo $info[id]:?>"> 
<?php echo $info[typename]:?></option> 
<?php }while($info=mysql fetch array($sqD):?> 
<2php 
$sql2=mysql query("select * from tb publishing”): 
$info2=mysql ftch_array($sql2): 
dof 
?> 
<option Value="<?php echo $info2[ISBN]:?>"> 
<?php echo $info2[pubname]:?></option> 
<?php }while($info2=mysqgl fetch array($sql2)):?> 
<?php 
$sql3=mysql_query("select * from tb_bookcase"); 
Sinfo3=mysql_fetch_array($sql3); 
dof 
> 
<option value="<?php echo $info3[id]:?>"> 
<?php echo $info3[name]:?></option> 


value="<?php echo $info3[name]:?>" 


图 书 类 型 


出 版 社 


书架 名 称 


操作 员 


Submit 


‘onClick="return check(forml)” 


“保存 ”按钮 


Submit2 


onClick="history.backO:” 


“返回 ”按钮 


由 于 添加 图 书信 息 的 方法 同 添加 管理 员 信 息 的 方法 类 似 ， 所 以 此 处 只 给 出 向 图 书信 息 表 中 插入 数据 的 
SQL 语句 ， 详 细 代码 参见 光盘 。 向 图 书信 息 表 中 插入 数据 的 SQL 语句 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\28\book_ok.php) 


mysql_query("insert into 


tb_bookinfo(barcode,bookName ,typeid,author,translator,ISBN ,price,page,bookcase,inTime,operator )values('$b 
arcode','$bookName','$typeid','$author','$translator','$isbn','$price','$page','$bookcaseid','$inTime','$operator’)"); 


28.7.5 ”修改 图 书信 息 页 面 的 实现 过 程 


管理 员 登 录 系统 后 ， 在 导航 栏 中 单 寺 


Ff “图 书 档案 管理 ” 超 链 接 ， 进 入 到 查看 图 书 列表 页 面 。 单 击 想 要 
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添加 图 书信 息 页 面 主要 用 于 收集 输入 的 图 书信 息 以 及 通过 自 定义 的 JavaScript 函数 验证 输入 信息 是 否 合 
法 。 该 页 面 中 所 涉及 到 的 重要 表单 元 素 如 表 28.4 所 示 。 
表 28.4 添加 图 书信 息 页 面 所 涉及 的 重要 表单 元 素 


重要 属性 


method="post" action= "book okphp” 


typeld 


isbn 


bookcaseid 


operator 


<?php include("Conn/conn.php"); 
$sql=mysql_query("select * from tb_booktype"); 
Sinfo=mysql fetch array($sq)); 
dof 
?> 
<option value="<?php echo $info[id]:?>"> 
<?php echo $info[typename]:?></option> 
<?php }while($info=mysql fetch array($sqD):?> 
<2php 
$sql2=mysql query("select * from tb publishing”): 
$info2=mysql ftch_array($sql2): 
dof 
?> 
<option Value="<?php echo $info2[ISBN]:?>"> 
<?php echo $info2[pubname]:?></option> 
<?php }while($info2=mysqgl fetch array($sql2)):?> 
<?php 
$sql3=mysql_query("select * from tb_bookcase"); 
Sinfo3=mysql_fetch_array($sql3); 
dof 
> 
<option value="<?php echo $info3[id]:?>"> 
<?php echo $info3[name]:?></option> 


value="<?php echo $info3[name]:?>" 


图 书 类 型 


出 版 社 


书架 名 称 


操作 员 


Submit 


‘onClick="return check(forml)” 


“保存 ”按钮 


Submit2 


onClick="history.backO:” 


“返回 ”按钮 


由 于 添加 图 书信 息 的 方法 同 添加 管理 员 信 息 的 方法 类 似 ， 所 以 此 处 只 给 出 向 图 书信 息 表 中 插入 数据 的 
SQL 语句 ， 详 细 代码 参见 光盘 。 向 图 书信 息 表 中 插入 数据 的 SQL 语句 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\28\book_ok.php) 


mysql_query("insert into 


tb_bookinfo(barcode,bookName ,typeid,author,translator,ISBN ,price,page,bookcase,inTime,operator )values('$b 
arcode','$bookName','$typeid','$author','$translator','$isbn','$price','$page','$bookcaseid','$inTime','$operator’)"); 


28.7.5 ”修改 图 书信 息 页 面 的 实现 过 程 


管理 员 登 录 系统 后 ， 在 导航 栏 中 单 寺 


Ff “图 书 档案 管理 ” 超 链 接 ， 进 入 到 查看 图 书 列表 页 面 。 单 击 想 要 
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修改 的 图 书信 息 后 面 的 “修改 ” 超 链 接 , 进入 到 修改 图 书信 息 页 面 。 修改 图 书信 息 页 面 的 运行 结果 如 图 28.26 


所 示 。 


图 28.26 ”修改 图 书信 息 页 面 的 运行 结果 
在 查看 图 书信 息 列 表 页 面 中 ， 添 加 “修改 ” 超 链 接 的 代码 如 下 : 


(代码 位 置 ， 光 盘 \TM\Instance\28\book_ok.php) 


<a href="book_Modify.php?id=<?php echo $resultfbookid];?>"> 修 改 </a> 

在 修改 图 书信 息 页 面 中 修改 图 书信 息 后 , 单 击 “ 保 存 ” 按 钮 ,提交 表单 信息 到 数据 处 理 页 book_Modify_ 
ok.php， 应 用 UPDATE 语句 将 修改 的 图 书信 息 保 存 到 数据 表 tb_bookinfo 中 ， 修 改 成 功 后 ， 则 弹出 “图 书信 
息 修改 成 功 ! ”提示 信息 ， 并 将 页 面 重 定向 到 修改 图 书信 息 页 。 数 据 处 理 页 的 代码 如 下 : 


(代码 位 置 ， 光 盘 \TM\Instance\28\book_Modify_ok.php) 


<?php 

session_start(); 
include("conn/conn.php"); 
$bid=$_POSTI[bid]; 
S$operator=$_SESSION[admin_name]; 
$barcode=$_POST[barcode]; 
S$bookName=$_POST[bookName]; 
Stypeid=$_POSTI[typeld]; 
$author=$_POST[author]; 
$translator=$_POST[translator]; 
$isbn=$_POST[isbn]; 
$price=$_POSTIprice]; 
$page=$_POSTIpage]; 
$bookcase=$_POST[bookcaseid]; 
S$inTime=date("Y-m-d"); 


// 初 始 化 session 变量 
// 连 接 数 据 库 文件 

// 获 取 图 书 id 号 

// 获 取 管理 员 名 称 

// 获 取 图 书 条 形 码 
/获取 图 书 名 称 

// 获 取 图 书 类 型 id 号 
// 获 取 图 书 作者 

// 获 取 图 书 译 者 

// 获 取出 版 社 ISBN 

// 获 取 图 书 单价 


/设置 图 书 更 新 日 期 为 当前 日 期 


$query=mysql_query("update tb_bookinfo set barcode='$barcode', bookName='$bookName' , typeid='$typeid', 
author='$author, translator='$translator, ISBN='$isbn' , price='$price' , page='$page' , bookcase='$bookcaseid', 


inTime='$inTime',, operator='$operator where id=$bid"); 


/更 新 数据 表 


echo "<scriptlanguage='javascript>alert( 图 书信 息 修改 成 功 ");history.back();</script>”; 


?> 


四 
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修改 的 图 书信 息 后 面 的 “修改 ” 超 链 接 , 进入 到 修改 图 书信 息 页 面 。 修改 图 书信 息 页 面 的 运行 结果 如 图 28.26 


所 示 。 


图 28.26 ”修改 图 书信 息 页 面 的 运行 结果 
在 查看 图 书信 息 列 表 页 面 中 ， 添 加 “修改 ” 超 链 接 的 代码 如 下 : 


(代码 位 置 ， 光 盘 \TM\Instance\28\book_ok.php) 


<a href="book_Modify.php?id=<?php echo $resultfbookid];?>"> 修 改 </a> 

在 修改 图 书信 息 页 面 中 修改 图 书信 息 后 , 单 击 “ 保 存 ” 按 钮 ,提交 表单 信息 到 数据 处 理 页 book_Modify_ 
ok.php， 应 用 UPDATE 语句 将 修改 的 图 书信 息 保 存 到 数据 表 tb_bookinfo 中 ， 修 改 成 功 后 ， 则 弹出 “图 书信 
息 修改 成 功 ! ”提示 信息 ， 并 将 页 面 重 定向 到 修改 图 书信 息 页 。 数 据 处 理 页 的 代码 如 下 : 


(代码 位 置 ， 光 盘 \TM\Instance\28\book_Modify_ok.php) 


<?php 

session_start(); 
include("conn/conn.php"); 
$bid=$_POSTI[bid]; 
S$operator=$_SESSION[admin_name]; 
$barcode=$_POST[barcode]; 
S$bookName=$_POST[bookName]; 
Stypeid=$_POSTI[typeld]; 
$author=$_POST[author]; 
$translator=$_POST[translator]; 
$isbn=$_POST[isbn]; 
$price=$_POSTIprice]; 
$page=$_POSTIpage]; 
$bookcase=$_POST[bookcaseid]; 
S$inTime=date("Y-m-d"); 


// 初 始 化 session 变量 
// 连 接 数 据 库 文件 

// 获 取 图 书 id 号 

// 获 取 管理 员 名 称 

// 获 取 图 书 条 形 码 
/获取 图 书 名 称 

// 获 取 图 书 类 型 id 号 
// 获 取 图 书 作者 

// 获 取 图 书 译 者 

// 获 取出 版 社 ISBN 

// 获 取 图 书 单价 


/设置 图 书 更 新 日 期 为 当前 日 期 


$query=mysql_query("update tb_bookinfo set barcode='$barcode', bookName='$bookName' , typeid='$typeid', 
author='$author, translator='$translator, ISBN='$isbn' , price='$price' , page='$page' , bookcase='$bookcaseid', 


inTime='$inTime',, operator='$operator where id=$bid"); 


/更 新 数据 表 


echo "<scriptlanguage='javascript>alert( 图 书信 息 修改 成 功 ");history.back();</script>”; 


?> 
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28.7.6 ”删除 图 书信 息 的 实现 过 程 


在 查看 图 书信 息 列表 页 面 中 ， 设 置 “ 删 除 ” 超 链接 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\28\book.php) 

<a href="book_del.php?id=<?php echo $resultfbookid];?>"> 删 除 </a> 

单 击 想 要 删除 的 图 书信 息 后 面 的 “删除 ” 超 链 接 ， 提 交 表 单 信息 到 数据 处 理 页 book_delphp， 应 用 
DELETE 语句 将 指定 的 图 书信 息 从 数据 表 tb_bookinfo 中 删除 ， 如 果 删 除 操作 执行 成 功 ， 则 弹出 “图 书信 息 
删除 成 功 ! ”提示 信息 ， 并 将 页 面 重 定向 到 图 书信 息 列 表 页 面 。 数 据 处 理 页 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\28\book_del.php) 


<?php 

include("conn/conn.php"); // 连 接 数据 库 文件 
$info_del=mysql_query("delete from tb_bookinfo where id=$_GETI[id]"); /删除 指定 的 图 书信 息 
if($info_delX{ // 如 果 信 息 删 除 成 功 , 则 弹出 提示 


echo "<script language='javascript>alert(' 图 书信 息 删 除 成 功 !");history.back();</script> "; 
} 


?> 


28.8 图 书 借 还 模块 设计 


峰 4 视频 讲解 : 光盘 \TM\Video\ 第 28 章 \ 图 书 借 还 模块 设计 .exe 
28.8.1 图 书 借 还 模块 概述 
图 书 借 还 模块 主要 包括 图 书 借阅 、 图 书 续 借 、 图 书 归 还 、 图 书 档案 查询 、 图 书 借阅 查询 、 借 阅 到 期 提 


醒 6 个 功能 。 在 图 书 借阅 模块 中 的 用 户 只 有 一 种 身份 ， 即 操作 员 ， 通 过 该 身份 可 以 进行 图 书 借 还 等 相关 操 
作 。 图 书 借 还 模块 的 结构 图 如 图 28.27 所 示 。 


图书 档 案 查 询 ~ 图 书 佛 闻 
CC - 9/ i 


pe Na, 
借阅 到 期 提醒 


图 28.27 图 书 借 还 模块 的 结构 图 
28.8.2 图 书 借 还 模块 中 的 多 表 查 询 技术 


在 图 书 借 还 模块 中 涉及 到 的 数据 表 是 tb_borow 〈 图 书 借阅 信息 表 ) 、tb_bookinfo 〈 图 书信 息 表 ) 和 
tb reader〈 读 者 信息 表 ) ， 这 3 个 数据 表 间 通过 相应 的 字段 进行 关联 ， 如 图 28.28 所 示 。 
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28.7.6 ”删除 图 书信 息 的 实现 过 程 


在 查看 图 书信 息 列表 页 面 中 ， 设 置 “ 删 除 ” 超 链接 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\28\book.php) 

<a href="book_del.php?id=<?php echo $resultfbookid];?>"> 删 除 </a> 

单 击 想 要 删除 的 图 书信 息 后 面 的 “删除 ” 超 链 接 ， 提 交 表 单 信息 到 数据 处 理 页 book_delphp， 应 用 
DELETE 语句 将 指定 的 图 书信 息 从 数据 表 tb_bookinfo 中 删除 ， 如 果 删 除 操作 执行 成 功 ， 则 弹出 “图 书信 息 
删除 成 功 ! ”提示 信息 ， 并 将 页 面 重 定向 到 图 书信 息 列 表 页 面 。 数 据 处 理 页 的 代码 如 下 : 

(代码 位 置 ， 光盘 \TM\Instance\28\book_del.php) 


<?php 

include("conn/conn.php"); // 连 接 数据 库 文件 
$info_del=mysql_query("delete from tb_bookinfo where id=$_GETI[id]"); /删除 指定 的 图 书信 息 
if($info_delX{ // 如 果 信 息 删 除 成 功 , 则 弹出 提示 


echo "<script language='javascript>alert(' 图 书信 息 删 除 成 功 !");history.back();</script> "; 
} 


?> 


28.8 图 书 借 还 模块 设计 


峰 4 视频 讲解 : 光盘 \TM\Video\ 第 28 章 \ 图 书 借 还 模块 设计 .exe 
28.8.1 图 书 借 还 模块 概述 
图 书 借 还 模块 主要 包括 图 书 借阅 、 图 书 续 借 、 图 书 归 还 、 图 书 档案 查询 、 图 书 借阅 查询 、 借 阅 到 期 提 


醒 6 个 功能 。 在 图 书 借阅 模块 中 的 用 户 只 有 一 种 身份 ， 即 操作 员 ， 通 过 该 身份 可 以 进行 图 书 借 还 等 相关 操 
作 。 图 书 借 还 模块 的 结构 图 如 图 28.27 所 示 。 
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pe Na, 
借阅 到 期 提醒 


图 28.27 图 书 借 还 模块 的 结构 图 
28.8.2 图 书 借 还 模块 中 的 多 表 查 询 技术 


在 图 书 借 还 模块 中 涉及 到 的 数据 表 是 tb_borow 〈 图 书 借阅 信息 表 ) 、tb_bookinfo 〈 图 书信 息 表 ) 和 
tb reader〈 读 者 信息 表 ) ， 这 3 个 数据 表 间 通过 相应 的 字段 进行 关联 ， 如 图 28.28 所 示 。 


@ 
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op: 
remark text 
tpsid inteeer (11) 


er 0) 
isek tinyint () 


28.28 ”图 书 借 还 模块 各 表 间 关系 图 
28.8.3 ”图 书 借阅 页 面 的 实现 过 程 


管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 借阅 ”命令 ， 进 入 到 图 书 借阅 页 面 。 在 该 页 面 中 的 “读者 条 
形 码 ” 文 本 框 中 输入 读者 的 条 形 码 (如 1234561789) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 读者 的 
基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检索 到 对 应 的 读者 信息 ， 将 其 显示 在 页 面 中 ， 此 时 输入 图 书 的 条 
形 码 或 图 书 名 称 后 ， 单 击 “ 确 定 ” 按 钮 ， 借 阅 指定 的 图 书 ， 运 行 结果 如 图 28.29 所 示 。 


LI 加 五 异 阅 

痢 胎 下 读 看 条 形 码 。 12355 而 3 

媳 ”五 ; 降 丽 性 别 ; 区 读者 基 型 ， 公务员 

证 作 关 型 : 机 证 证 从: UU 可 仙 外 县: 后 

添加 的 六 妖 : 人” 图书 条 形 码 。。 C 图 书包 各 5 二 EL 
图 书 名 称 Le REg 间 [ 全 从) 

PIE 国 半 地 性 大 全 2007-12-06 2008-01-05 人 民有 邮电 出 版 社 PE 书架 .0 

Fe 和 指 库 系 寺 开发 完全 手 用 200-12-08 | 2008-01-05 人 民 才 中 册 版 村 3 EE 


图 28.29 图 书 借阅 页 面 的 运行 结果 


SG 当 读 者 借阅 图 书 完毕 后 ， 单 去 “完成 借阅 ”按钮 ， 将 重新 载 入 图 书 借阅 页 面 ， 当 前 页 处 于 空 信 
息 状态 ， 从 而 方便 操作 员 进 行 下 一 个 读者 借阅 图 书 的 操作 。 


图 书 借阅 页 面 可 以 分 为 两 部 分 : 一 部 分 用 于 查询 并 显示 读者 信息 ;， 另 一 部 分 用 于 显示 和 添加 读者 的 借 
阅 信息 。 图 书 借阅 页 面 在 Dreamweaver 中 的 设计 效果 如 图 28.30 所 示 。 


图 28.30 图书 借 阅 页 面 的 设计 效果 
在 进行 图 书 借阅 时 ， 系 统 要 求 每 个 读者 只 能 同时 借阅 一 定数 量 的 图 书 ， 并 且 该 数量 由 读者 类 型 表 


tb readerType 中 的 可 借 数 量 number 决定 ， 所 以 笔者 编写 了 自 定义 的 checkbook0 函 数 , 用 于 判断 当前 选择 的 
读者 是 否 还 可 以 借阅 新 的 图 书 ， 同 时 该 函数 还 具有 判断 读者 条 形 码 或 图 书 名 称 文本 框 是 否 为 空 的 功能 。 其 


代码 如 下 : 
DD 


(代码 位 置 ， 光盘 \TM\Instance\28\book_del.php) 
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<script language="javascript"> 


function checkbook(formX{ // 自 定义 一 个 JavaScript 函数 checkbook() 
ifform.barcode value==""){ // 如 果 读 者 条 形 码 为 空 
alert(" 请 输入 读者 条 形 码 ");form.barcode.focus();return; // 弹 出 提示 ， 焦 点 返回 到 条 形 码 文本 框 
} 
if(form.inputkey.value=="™"}{ // 如 果 图 书 查询 文本 框 的 值 为 空 
alert(" 请 输入 查询 关键 字 ");form.inputkey.focus();return; // 弹 出 提示 ， 焦 点 返回 到 图 书 查询 文本 框 
i 
if(form.number.value-form.borrowNumber.value<=0}{ /如果 图 书 的 借阅 数量 超过 了 可 借 数量 
alert(" 您 不 能 再 借阅 其 他 图 书 了 ");return; // 弹 出 提示 信息 
b 
form.submit(); // 提 交 表单 
</script> 


有 


[a 技巧 在 JavaScript 中 比较 两 个 数值 型 文本 框 的 值 时 ， 不 使 用 运算 符 “==”， 而 是 将 这 两 个 值 相 减 ， 
再 判断 其 结果 。 


检索 读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 的 SQL 语句 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\28\bookborrow.php) 
$sql=mysql_query("select r.*,t.name as typename,tnumber from tb_reader r left join tb_readerType t on 
rtypeid=tid where rbarcode='$barcode'"); 
$info=mysql_fetch_array($sql); /检索 读者 信息 
获取 读者 借阅 信息 的 SQL 语句 如 下 : 
《代码 位 置 ， 光 盘 \TM\Instance\28\bookborrow.php) 
$sql1=mysql_query("select r.*,borr.borrowTime,borr.backTime,book.bookname,book.price,pub.pubname,bc.name 
as bookcase from tb_borrow as borr join tb_bookinfo as book on book.id=borr.bookid join tb_publishing as pub 
on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id join tb_reader as r on 
borr.readerid=r.id where borr.readerid='$readerid' and borr.ifback=0"); 
$info1=mysql_fetch_array($sql1); /检索 读者 的 借阅 信息 
$borrowNumber=mysql_num_rows($sql1); // 获 取 结 果 集中 行 的 数目 
在 “图 书 条 形 码 ”或 “图 书 名 称 ” 文 本 框 中 输入 图 书 条 形 码 或 图 书 名 称 后 ， 单 击 “确定 ”按钮 ， 检 索 
图 书信 息 是 否 存在 ， 如 果 不 存 在 ， 则 向 图 书 借阅 信息 表 中 添加 该 读者 的 图 书 借阅 记录 ， 完 成 图 书 的 借阅 操 
作 ， 和 否则， 弹出 该 图 书 不 能 被 同一 读者 重复 借阅 的 提示 信息 。 图 书 借阅 的 具体 代码 如 下 : 
(代码 位 置 ， 光盘 \TMNInstance\28\bookBorrow.php) 


<?php 

if($_POST[inputkey]!="}{ // 如 果 “ 图 书 条 形 码 ”/ “图 书 名 称 ” 后 的 文本 框 不 为 空 
$f=$_POSTIN; // 获 取 用 户 选择 的 条 件 值 

@ $inputkey=trim($_POST[inputkey]); 1/ 获取 用 户 输入 的 查询 关键 字 
$barcode=$_POST[barcode]; 1/ 获取 读者 的 条 形 码 

$readerid=$_POST[readerid]; // 获 取 读 者 id 号 

日 $borrowTime=date(Y-m-d); /图 书 的 借阅 时 间 为 系统 当前 时 间 


@ ”$backTime=date("Y-m-d",(time()+3600*24*30)); /归还 图 书 日 期 为 当前 日 期 +30 天 期 限 
$query=mysql_query("select * from tb_bookinfo where $f='$inputkey"); 


Sresult=mysql_fetch_array($query); // 检 索 图 书信 息 是 否 存 在 
if($result==falseX{ // 如 果 读 者 借阅 的 图 书 不 存在 ， 那 么 弹出 提示 信息 


echo "<script language='javascript>alert(' 该 图 书 不 存在 ! ');window.location.href='"bookBorrow.php? barcode= 
$barcode'; </script>"; 


有 
else{ /检索 该 读者 所 借阅 的 图 书 是 否 与 再 借 图 书 重复 
$query1=mysql_query("select r*,borr.borrowTime,borr.backTime,book.bookname,book.price,pub.pubname, 
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<script language="javascript"> 


function checkbook(formX{ // 自 定义 一 个 JavaScript 函数 checkbook() 
ifform.barcode value==""){ // 如 果 读 者 条 形 码 为 空 
alert(" 请 输入 读者 条 形 码 ");form.barcode.focus();return; // 弹 出 提示 ， 焦 点 返回 到 条 形 码 文本 框 
} 
if(form.inputkey.value=="™"}{ // 如 果 图 书 查询 文本 框 的 值 为 空 
alert(" 请 输入 查询 关键 字 ");form.inputkey.focus();return; // 弹 出 提示 ， 焦 点 返回 到 图 书 查询 文本 框 
i 
if(form.number.value-form.borrowNumber.value<=0}{ /如果 图 书 的 借阅 数量 超过 了 可 借 数量 
alert(" 您 不 能 再 借阅 其 他 图 书 了 ");return; // 弹 出 提示 信息 
b 
form.submit(); // 提 交 表单 
</script> 


有 


[a 技巧 在 JavaScript 中 比较 两 个 数值 型 文本 框 的 值 时 ， 不 使 用 运算 符 “==”， 而 是 将 这 两 个 值 相 减 ， 
再 判断 其 结果 。 


检索 读者 的 基本 信息 和 未 归还 的 借阅 图 书信 息 的 SQL 语句 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\28\bookborrow.php) 
$sql=mysql_query("select r.*,t.name as typename,tnumber from tb_reader r left join tb_readerType t on 
rtypeid=tid where rbarcode='$barcode'"); 
$info=mysql_fetch_array($sql); /检索 读者 信息 
获取 读者 借阅 信息 的 SQL 语句 如 下 : 
《代码 位 置 ， 光 盘 \TM\Instance\28\bookborrow.php) 
$sql1=mysql_query("select r.*,borr.borrowTime,borr.backTime,book.bookname,book.price,pub.pubname,bc.name 
as bookcase from tb_borrow as borr join tb_bookinfo as book on book.id=borr.bookid join tb_publishing as pub 
on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id join tb_reader as r on 
borr.readerid=r.id where borr.readerid='$readerid' and borr.ifback=0"); 
$info1=mysql_fetch_array($sql1); /检索 读者 的 借阅 信息 
$borrowNumber=mysql_num_rows($sql1); // 获 取 结 果 集中 行 的 数目 
在 “图 书 条 形 码 ”或 “图 书 名 称 ” 文 本 框 中 输入 图 书 条 形 码 或 图 书 名 称 后 ， 单 击 “确定 ”按钮 ， 检 索 
图 书信 息 是 否 存在 ， 如 果 不 存 在 ， 则 向 图 书 借阅 信息 表 中 添加 该 读者 的 图 书 借阅 记录 ， 完 成 图 书 的 借阅 操 
作 ， 和 否则， 弹出 该 图 书 不 能 被 同一 读者 重复 借阅 的 提示 信息 。 图 书 借阅 的 具体 代码 如 下 : 
(代码 位 置 ， 光盘 \TMNInstance\28\bookBorrow.php) 


<?php 

if($_POST[inputkey]!="}{ // 如 果 “ 图 书 条 形 码 ”/ “图 书 名 称 ” 后 的 文本 框 不 为 空 
$f=$_POSTIN; // 获 取 用 户 选择 的 条 件 值 

@ $inputkey=trim($_POST[inputkey]); 1/ 获取 用 户 输入 的 查询 关键 字 
$barcode=$_POST[barcode]; 1/ 获取 读者 的 条 形 码 

$readerid=$_POST[readerid]; // 获 取 读 者 id 号 

日 $borrowTime=date(Y-m-d); /图 书 的 借阅 时 间 为 系统 当前 时 间 


@ ”$backTime=date("Y-m-d",(time()+3600*24*30)); /归还 图 书 日 期 为 当前 日 期 +30 天 期 限 
$query=mysql_query("select * from tb_bookinfo where $f='$inputkey"); 


Sresult=mysql_fetch_array($query); // 检 索 图 书信 息 是 否 存 在 
if($result==falseX{ // 如 果 读 者 借阅 的 图 书 不 存在 ， 那 么 弹出 提示 信息 


echo "<script language='javascript>alert(' 该 图 书 不 存在 ! ');window.location.href='"bookBorrow.php? barcode= 
$barcode'; </script>"; 


有 
else{ /检索 该 读者 所 借阅 的 图 书 是 否 与 再 借 图 书 重复 
$query1=mysql_query("select r*,borr.borrowTime,borr.backTime,book.bookname,book.price,pub.pubname, 
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bc.name as bookcase from tb_borrow as borr join tb_reader as r on borr.readerid=r.id join tb_bookinfo as book 
on book.id=borr.bookid join tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on 
book.bookcase=bc.id ”where borr.bookid=$result[id] and borr.readerid=$readerid and ifoack=0"); 
Sresult1=mysql_fetch_array($query1); 
if($result1==trueX{ // 如 果 所 借 图 书 已 被 该 读者 借阅 ， 那 么 提示 不 能 重复 借阅 
echo "<script language='javascript>alert( 该 图 书 已 经 借阅 ! ');window.location.href='bookBorrow.php? 
barcode=$barcode';</script>"; 
} 
else{ /否则 ， 完 成 图 书 借阅 操作 ， 并 弹出 借阅 成 功 提示 信息 
$bookid=S$resultlid]; /将 读者 id 号 赋 给 一 变量 
mysql_query("insert into 
tb_borrow(readerid,bookid,borrowTime,backTime,operator,ifback)values('$readerid','$bookid','$borrowTime','$b 
ackTime','$_SESSION[admin_name]',0)"); // 向 借阅 信息 表 中 添加 一 条 借阅 信息 
echo "<script language='javascript>alert(' 图 书 借阅 操作 成 功 ! ');window.location.href='bookBorrow.php? 
barcode=$barcode';</script>"; 
} 


} 


?> 


SC 人 
@tim0: 删除 字符 串 中 首尾 的 空格 或 者 其 他 字符 ， 以 达到 精确 查询 。 
@ date(Y-m-d): 获取 系统 的 当前 日 期 为 图 书 借阅 的 日 期 ， 并 格式 化 日 期 格式 。 
@ date("Y-m-d".(time0+3600*24*30)): 归还 图 书 日 期 = 当前 日 期 130 天 期 限 .当前 日 期 时 间 戳 应 用 time0 
函数 获取 ，30 天 期 限 的 时 间 蕉 等 于 3600 秒 x24 小 时 x30 天 ， 并 通过 date0 函 数 格式 化 为 指定 日 期 格式 。 


28.8.4 图 书 续 借 页 面 的 实现 过 程 


管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 续 借 ”命令 ， 进 入 到 图 书 续 借 页 面 。 在 该 页 面 中 的 “读者 条 
形 码 ”文本 框 中 输入 读者 的 条 形 码 (如 1234561789) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 读者 的 
基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检索 到 对 应 的 读者 信息 ， 则 将 其 显示 在 页 面 中 ， 此 时 单 击 “ 续 借 ” 
超 链接 ， 即 可 续 借 指定 图 书 〈 即 将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 ) 。 图 书 续 借 页 面 
的 运行 结果 如 图 28.31 所 示 。 
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图 28.31 图 书 续 借 页 面 的 运行 结果 
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bc.name as bookcase from tb_borrow as borr join tb_reader as r on borr.readerid=r.id join tb_bookinfo as book 
on book.id=borr.bookid join tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on 
book.bookcase=bc.id ”where borr.bookid=$result[id] and borr.readerid=$readerid and ifoack=0"); 
Sresult1=mysql_fetch_array($query1); 
if($result1==trueX{ // 如 果 所 借 图 书 已 被 该 读者 借阅 ， 那 么 提示 不 能 重复 借阅 
echo "<script language='javascript>alert( 该 图 书 已 经 借阅 ! ');window.location.href='bookBorrow.php? 
barcode=$barcode';</script>"; 
} 
else{ /否则 ， 完 成 图 书 借阅 操作 ， 并 弹出 借阅 成 功 提示 信息 
$bookid=S$resultlid]; /将 读者 id 号 赋 给 一 变量 
mysql_query("insert into 
tb_borrow(readerid,bookid,borrowTime,backTime,operator,ifback)values('$readerid','$bookid','$borrowTime','$b 
ackTime','$_SESSION[admin_name]',0)"); // 向 借阅 信息 表 中 添加 一 条 借阅 信息 
echo "<script language='javascript>alert(' 图 书 借阅 操作 成 功 ! ');window.location.href='bookBorrow.php? 
barcode=$barcode';</script>"; 
} 


} 


?> 


SC 人 
@tim0: 删除 字符 串 中 首尾 的 空格 或 者 其 他 字符 ， 以 达到 精确 查询 。 
@ date(Y-m-d): 获取 系统 的 当前 日 期 为 图 书 借阅 的 日 期 ， 并 格式 化 日 期 格式 。 
@ date("Y-m-d".(time0+3600*24*30)): 归还 图 书 日 期 = 当前 日 期 130 天 期 限 .当前 日 期 时 间 戳 应 用 time0 
函数 获取 ，30 天 期 限 的 时 间 蕉 等 于 3600 秒 x24 小 时 x30 天 ， 并 通过 date0 函 数 格式 化 为 指定 日 期 格式 。 


28.8.4 图 书 续 借 页 面 的 实现 过 程 


管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 续 借 ”命令 ， 进 入 到 图 书 续 借 页 面 。 在 该 页 面 中 的 “读者 条 
形 码 ”文本 框 中 输入 读者 的 条 形 码 (如 1234561789) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 读者 的 
基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检索 到 对 应 的 读者 信息 ， 则 将 其 显示 在 页 面 中 ， 此 时 单 击 “ 续 借 ” 
超 链接 ， 即 可 续 借 指定 图 书 〈 即 将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 ) 。 图 书 续 借 页 面 
的 运行 结果 如 图 28.31 所 示 。 
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?起 汪 析 多才 大 全 rieo ares AR eiR 可 om 和 
Pe 才 所 放 系 统 开导 全 手 负 。 2207-t2-06 。 2nce ol-o5 。 人 RN 守 。 PP 书 于 em 


CayHidt e e007 wer wtecd em 吉村 "re 打 王 大 学 入 
本 庆 有 全 IE 6 0 或 内 上 白 本 IDesrTas 为 县 信 示 过 时 


图 28.31 图 书 续 借 页 面 的 运行 结果 
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/ 
和 吉明 当 读者 续 借 完 图 书后 ， 单 击 “ 完 成 续 借 ”按钮 ， 将 重新 载 入 图 书 续 借 页 面 ， 当 前 页 处 于 空 信 息 
状态 ， 从 而 方便 操作 员 进 行 下 一 个 读者 续 借 图 书 的 操作 。 

图 书 续 借 页 面 的 设计 方法 同 图 书 借阅 页 面 类 似 ， 不 同 的 是 ， 在 图 书 续 借 页 面 中 没有 添加 借阅 图 书 的 功 
图 书 续 借 页 面 在 Dreamweaver 中 的 设计 效果 如 图 28.32 所 示 。 
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图 28.32 图书 续 借 页 面 的 设计 效果 

单 击 “ 续 借 ” 超 链接 时 ， 还 需要 将 读者 条 形 码 、 借 阅 id 号 和 图 书 归还 时 间 一 同 传递 到 图 书 续 借 的 处 理 
页 borrow_oncemore.php 中 。 其 代码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\28\bookRenew.php) 

<a href="borrow_oncemore.php?barcode=<?php echo $info[barcode];?>&borrid=<?php echo 

S$info[borrid];?>&backTime=<?php echo $info[backTime];?>"> 续 借 </a> 

检索 读者 信息 和 读者 借阅 信息 的 SQL 语句 如 下 : 

(代码 位 置 : 光盘 \TM\Instance\28\bookRenew.php ) 

$sql=mysql_query("select borrid as borrid,borr.borrowTime,borr.backTime,borr.ifoack,r.*t.name as 

typename,tnumber,book.bookname,book.price,pub.pubname,bc.name as bookcase from tb_borrow as borr 

join tb_reader r on borr.readerid=r.id join tb_readerType t on r.typeid=t.id join tb_bookinfo as book on 

book.id=borr.bookid join tb_publishing as pub on book.ISBN=pub.ISBN join tb_bookcase as bc on 

book.bookcase=bc.id where r.barcode='$barcode' and borr.ifback=0"); 

Sinfo=mysql_fetch_array($sql); // 检 索 读 者 信息 和 借阅 信息 

单 击 “ 续 借 ” 超 链接 ， 提 交 到 数据 处 理 页 borrow_oncemore.php， 用 于 完成 图 书 的 续 借 功 能 ， 主 要 通过 
更 改 图 书 的 归还 日 期 (即将 该 图 书 的 归还 时 间 加 上 该 书 的 可 借 天 数 30 天 计算 得 出 ) 实现。 数据 处 理 页 的 代 
码 如 下 : 

(代码 位 置 ， 光 盘 \TM\Instance\28\borrow_oncemore.php) 


<?php 

session_start(); /初始 化 session 变量 
include("conn/conn.php"); // 连 接 数据 库 文件 
S$barcode=$_GET[barcode]; 1/ 获取 图 书 条 形 码 
$new=$_GET[backTime]; /获取 图 书 归 还 时 间 


// 更 新 续 借 期 ， 将 动态 获取 的 还 书 日 期 转化 为 时 间 戳 ， 然 后 再 求 出 续 借 后 的 还 书 日 期 
$newbackTime=date("Y-m-d",(mktime(0, 0, 0, substr($new,5,2), substr($new,8,2), substr($new,0,4))+3600*24*30)); 
$borrid=$_GET[borrid]; // 获 取 续 借 图 书 的 id 号 
mysql_query("update tb_borrow set backTime='$newbackTime ,ifback=0,operator='$_SESSION[admin_name]' 
Where id=$borrid"); 

echo "<script language='javascript>alert(' 图 书 续 借 操 作成 功 ! ');window.location.href='bookRenew.php? 
barcode=$barcode';</script> // 弹 出 图 书 续 借 成 功 的 提示 信息 


?> 


他 
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28.8.5 ”图书 归还 页 面 的 实现 过 程 


管理 员 登 录 后 ， 选 择 “ 图 书 借 还 ”/“ 图 书 归 还 ”命令 ， 进 入 到 图 书 归 还 页 面 。 在 该 页 面 中 的 “读者 条 
形 码 ”文本 框 中 输入 读者 的 条 形 码 (如 1234561789) 后 ， 单 击 “ 确 定 ” 按 钮 ， 系 统 会 自动 检索 出 该 读者 的 
基本 信息 和 未 归还 的 借阅 图 书信 息 。 如 果 检索 到 对 应 的 读者 信息 ， 则 将 其 输出 到 浏览 器 ， 此 时 单 击 “ 归 还 ” 
超 链 接 ， 即 可 将 指定 图 书 归 还 。 图 书 归 还 页 面 的 运行 结果 如 图 28.33 所 示 。 
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图 28.33 图 书 归 还 页 面 的 运行 结果 


图 书 归还 页 面 的 设计 方法 同 图 书 续 借 页 面 类 似 ， 不 同 的 是 ， 将 图 书 续 借 页 面 中 的 “ 续 借 ” 超 链接 转化 
为 “归还 ” 超 链 接 。 在 单 击 “ 归 还 ” 超 链 接 时 ， 也 需要 将 读者 条 形 码 和 借阅 id 号 一 同 传递 到 图 书 归还 处 理 
页 。 其 代码 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\28\bookBack.php) 
<a href="bookBack_ok.php?borrid=<?php echo S$info[borrid];?>&barcode=<?php echo $info[barcode];?>"> 归 还 </a> 
检索 读者 信息 及 读者 借阅 信息 的 SQL 语句 如 下 : 
(代码 位 置 ， 光盘 \TM\Instance\28\bookBack.php) 
$sql=mysql_query("select borr.id as borrid,borr.borrowTime,borr.backTime,borr.ifoack,r.*,t.name as typename, 
tnumber,book.bookname,book.price,pub.pubname,bc.name as bookcase from tb_borrow as borr join tb_reader 
ron borr.readerid=r.id join tb_readerType t on r.typeid=t.id join tb_bookinfo as book on book.id=borr.bookid join 
tb_publishing as 
pub on book.ISBN=pub.ISBN join tb_bookcase as bc on book.bookcase=bc.id where rbarcode='$barcode' 
and borr.ifback=0"); 
$info=mysql_fetch_array($sql); /检索 读者 信息 及 该 读者 的 借阅 信息 
单 击 “ 归 还 ” 超 链接 ， 即 可 将 指定 图 书 归还 。 数 据 处 理 页 的 代码 如 下 : 
(代码 位 置 ， 光 盘 \TM\Instance\28\bookBack_ok.php) 


<?php 

session_start(); /| 初始 化 session 变量 
include("conn/conn.php"); // 连 接 数据 库 文件 
$backTime=date("Y-m-d"); // 归 还 图 书 日 期 
$borrid=$_GET[borrid]; // 获 取 读 者 的 id 号 
mysql_query("update tb_borrow set backTime='$backTime ,ifback=1,operator='$_SESSION[admin_name]' 
where id=$borrid"); // 更 新 读者 的 借阅 信息 


echo "<script language='javascript'>alert(' 图 书 归 还 操作 成 功 ! 
');window.location.href='bookBack.php?barcode=$barcode';</script>";， ”/// 弹 出 图 书 归 还 成 功 的 提示 信息 
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28.8.6 图书 借阅 查询 页 面 的 实现 过 程 


管理 员 登 录 后 ， 选 择 “系统 查询 ”/“ 图 书 借阅 查询 ”命令 ， 进 入 到 图 书 借阅 查询 页 面 。 图 书 借阅 查询 
页 面 的 运行 结果 如 图 28.34 所 示 。 在 该 页 面 中 可 以 按 指 定 的 字段 或 某 一 时 间 段 进行 查询 ， 同 时 还 可 以 实现 按 
指定 字段 及 时 间 段 进行 综合 条 件 查询 。 
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训 认 再 使 用 IE 6. 0 到 以 上 版 本 1024766 为 最 住 显示 效果 


28.34 ”图 书 借阅 查询 页 面 的 运行 结果 


图 书 借阅 查询 页 面 主 要 用 于 收集 查询 条 件 和 显示 查询 结果 ， 并 通过 自 定义 的 JavaScript 函数 验证 输入 的 
查询 条 件 是 否 合法 。 该 页 面 所 涉及 到 的 表单 元 素 如 表 28.5 所 示 。 


表 28.5 ”图书 借 阅 查 询 页 面 所 涉及 的 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含 义 

myform, form method="post" action="" 表单 

flagl checkbox value="a" 请 选择 查询 依据 

flag checkbox value="b" 借阅 时 间 
<option value="k.barcode" > 图 书 条 形 码 </option> 
<option value="k.bookname"> 图 书 名 称 </option> 

f a 0 i 这 六 中 妥 
<option value="r.name"> 读 者 名 称 </option> 

ke text Size="50" 关键 字 

sdate text id="sdate” 开始 日 期 

edate text id="edate" 结束 日 期 

Submit submit onClick="return check(myform):" “查询 ”按钮 

在 图 书 借阅 查询 页 面 中 ， 指 定 查询 条 件 后 ， 提 交 表 单 信息 到 当前 页 。 首 先 获取 表单 元 素 复 选 框 fag 的 


值 ， 然 后 根据 flag 的 值 组 合 查询 字符 串 。 如 果 flagl 的 值 等 于 a， 那么 按 指 定 的 字段 检索 图 书 借阅 信息 ， 如 
果 flag2 的 值 等 于 b, 那么 按 指定 的 时 间 段 检索 图 书 借阅 信息 ; 如 果 flagl 的 值 等 于 a, 并 且 flag2 的 值 等 于 b， 


那么 按 以 上 两 个 条 件 的 综合 条 件 检索 图 书 借阅 信息 ， 


(代码 位 置 ， 光盘 \TM\Instance\28\borrowQuery.php) 


GG 


并 将 查询 结果 输出 到 浏览 器 。 其 具体 代码 如 下 : 
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28.8.6 图书 借阅 查询 页 面 的 实现 过 程 


管理 员 登 录 后 ， 选 择 “系统 查询 ”/“ 图 书 借阅 查询 ”命令 ， 进 入 到 图 书 借阅 查询 页 面 。 图 书 借阅 查询 
页 面 的 运行 结果 如 图 28.34 所 示 。 在 该 页 面 中 可 以 按 指 定 的 字段 或 某 一 时 间 段 进行 查询 ， 同 时 还 可 以 实现 按 
指定 字段 及 时 间 段 进行 综合 条 件 查询 。 


是 可 还 
未 I3 环 
BT 
Beg 


他 置 : 区 从 进 图书 售 风 二 

[四 加 入 网 坦 询 

SS |y 请 过 拌 吉 调 俯 据 : ”区 下 名 补 。 。 ij0 开 六 战 1200 信 | EH]| 

局 二 二 以 [201-01-2 | 到 EL | 让 可 为: 2001-j2-01) 
图 代码 诬 看 亲 化 到 。。 志 汝 名 和 信 5 则 。。 Ji8 间 
123456789 PH 123455799 陈 十 2007-12-06 2006-01-05 
129450709 lzotamoe 褒 而 200r-12-06 2001-12-00 
123454321 PH 12355739 陈 丽 2011-01-25 2011-01-26 
corgi ht 6 2007 ww erbeet con 吉 机 ore 岳 范 大 学 图 所 究 
训 认 再 使 用 IE 6. 0 到 以 上 版 本 1024766 为 最 住 显示 效果 


28.34 ”图 书 借阅 查询 页 面 的 运行 结果 


图 书 借阅 查询 页 面 主 要 用 于 收集 查询 条 件 和 显示 查询 结果 ， 并 通过 自 定义 的 JavaScript 函数 验证 输入 的 
查询 条 件 是 否 合法 。 该 页 面 所 涉及 到 的 表单 元 素 如 表 28.5 所 示 。 


表 28.5 ”图书 借 阅 查 询 页 面 所 涉及 的 表单 元 素 


名 称 | 元 素 类 型 重要 属性 含 义 

myform, form method="post" action="" 表单 

flagl checkbox value="a" 请 选择 查询 依据 

flag checkbox value="b" 借阅 时 间 
<option value="k.barcode" > 图 书 条 形 码 </option> 
<option value="k.bookname"> 图 书 名 称 </option> 

f a 0 i 这 六 中 妥 
<option value="r.name"> 读 者 名 称 </option> 

ke text Size="50" 关键 字 

sdate text id="sdate” 开始 日 期 

edate text id="edate" 结束 日 期 

Submit submit onClick="return check(myform):" “查询 ”按钮 

在 图 书 借阅 查询 页 面 中 ， 指 定 查询 条 件 后 ， 提 交 表 单 信息 到 当前 页 。 首 先 获取 表单 元 素 复 选 框 fag 的 


值 ， 然 后 根据 flag 的 值 组 合 查询 字符 串 。 如 果 flagl 的 值 等 于 a， 那么 按 指 定 的 字段 检索 图 书 借阅 信息 ， 如 
果 flag2 的 值 等 于 b, 那么 按 指定 的 时 间 段 检索 图 书 借阅 信息 ; 如 果 flagl 的 值 等 于 a, 并 且 flag2 的 值 等 于 b， 


那么 按 以 上 两 个 条 件 的 综合 条 件 检索 图 书 借阅 信息 ， 


(代码 位 置 ， 光盘 \TM\Instance\28\borrowQuery.php) 


GG 


并 将 查询 结果 输出 到 浏览 器 。 其 具体 代码 如 下 : 
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<?php 

include("conn/conn.php"); /J 连接 数据 库 文件 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,r.barcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id"); 


// 查 询 图 书 借阅 信息 
if($_POSTISubmit]ll="™"X{ // 如 果 提 交 了 表单 ， 则 执行 以 下 操作 
$f=$_POSTIN:; // 获 取 操 作 员 选择 的 查询 条 件 

Skey1=$_POSTIkey1]; /获取 查询 关键 字 
$sdate=$_POST[sdate]; /获取 借阅 的 起 始 日 期 
Sedate=$_POST[edate]; // 获 取 借 阅 的 结束 日 期 
Sflag1=$_POSTIfag1]; /获取 按 指定 条 件 查询 的 复 选 框 值 
$flag2=$_POST[fNag2]; // 获 取 按 日 期 查询 的 复 选 框 值 


if($flag1=="a"}{ // 如 果 按 指定 条 件 查询 ， 则 执行 以 下 语句 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,r.barcode as readerbarcode,r.name,k.id,k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id where $f 
like '%$key1%"); 

. 


// 如 果 按时 间 段 查询 ， 则 执行 以 下 语句 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where borrowTime between '$sdate' and '$edate'"); 


} 

if($flag1=="a" && $flag2=="b"){ // 如 果 按 综合 条 件 查询 ， 则 执行 以 下 语句 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where borrowTime between '$sdate' and '$edate' and $f like '%$key1%"); 

} 


$result=mysql_fetch_array($sql); /检索 查询 结果 
if($result==false}{ // 如 果 查 询 结果 不 存在 ， 则 弹出 提示 信息 
?> 
<table width="100%" height="30" border="0" cellpadding="0" cellspacing="0"> 

<tr> 

<td height="36" align="center"> 暂 无 图 书 借阅 信息 ! </td> 

</tr> 
</table> 
<?php 
}else{ // 否 则 ， 输 出 图 书 借阅 信息 
?> 


<table width="1723"” border="1" cellpadding="0" cellspacing="0" bordercolor="#FFFFFF" bordercolordark= 
"#D2E3E6" bordercolorlight="#FFFFFF"> 
<tr align="center" bgcolor="#D0OE9F8"> 
<td width="13%"> 图 书 条 形 码 </td> 
<td width="217%"> 图 书 名 称 </td> 
<td width="15%"> 读 者 条 形 码 </td> 
<td width="11%"> 读 者 名 称 </td> 
<td width="13%"> 借 阅 时 间 </td> 
<td width="11%"> 归 还 时 间 </td> 
<td width="10%"> 是 否 归还 </td> 
</tr> 
<?php 
dof 
if($result[lifoack]=="0"}{ // 如 果 “ 是 否 归还 ”等 于 0, 则 输出 “未 归还 ” 


Sifbackstr=" 未 归还 "; 
@ 
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<?php 

include("conn/conn.php"); /J 连接 数据 库 文件 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,r.barcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id"); 


// 查 询 图 书 借阅 信息 
if($_POSTISubmit]ll="™"X{ // 如 果 提 交 了 表单 ， 则 执行 以 下 操作 
$f=$_POSTIN:; // 获 取 操 作 员 选择 的 查询 条 件 

Skey1=$_POSTIkey1]; /获取 查询 关键 字 
$sdate=$_POST[sdate]; /获取 借阅 的 起 始 日 期 
Sedate=$_POST[edate]; // 获 取 借 阅 的 结束 日 期 
Sflag1=$_POSTIfag1]; /获取 按 指定 条 件 查询 的 复 选 框 值 
$flag2=$_POST[fNag2]; // 获 取 按 日 期 查询 的 复 选 框 值 


if($flag1=="a"}{ // 如 果 按 指定 条 件 查询 ， 则 执行 以 下 语句 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,r.barcode as readerbarcode,r.name,k.id,k.barcode, 
k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id where $f 
like '%$key1%"); 

. 


// 如 果 按时 间 段 查询 ， 则 执行 以 下 语句 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where borrowTime between '$sdate' and '$edate'"); 


} 

if($flag1=="a" && $flag2=="b"){ // 如 果 按 综合 条 件 查询 ， 则 执行 以 下 语句 

$sql=mysql_query("select b.borrowTime,b.backTime,b.ifback,rbarcode as readerbarcode,r.name,k.id, 
k.barcode,k.bookname from tb_borrow b join tb_reader r on b.readerid=r.id join tb_bookinfo k on b.bookid=k.id 
where borrowTime between '$sdate' and '$edate' and $f like '%$key1%"); 

} 


$result=mysql_fetch_array($sql); /检索 查询 结果 
if($result==false}{ // 如 果 查 询 结果 不 存在 ， 则 弹出 提示 信息 
?> 
<table width="100%" height="30" border="0" cellpadding="0" cellspacing="0"> 

<tr> 

<td height="36" align="center"> 暂 无 图 书 借阅 信息 ! </td> 

</tr> 
</table> 
<?php 
}else{ // 否 则 ， 输 出 图 书 借阅 信息 
?> 


<table width="1723"” border="1" cellpadding="0" cellspacing="0" bordercolor="#FFFFFF" bordercolordark= 
"#D2E3E6" bordercolorlight="#FFFFFF"> 
<tr align="center" bgcolor="#D0OE9F8"> 
<td width="13%"> 图 书 条 形 码 </td> 
<td width="217%"> 图 书 名 称 </td> 
<td width="15%"> 读 者 条 形 码 </td> 
<td width="11%"> 读 者 名 称 </td> 
<td width="13%"> 借 阅 时 间 </td> 
<td width="11%"> 归 还 时 间 </td> 
<td width="10%"> 是 否 归还 </td> 
</tr> 
<?php 
dof 
if($result[lifoack]=="0"}{ // 如 果 “ 是 否 归还 ”等 于 0, 则 输出 “未 归还 ” 


Sifbackstr=" 未 归还 "; 
@ 
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Jelse{ 1/ 如果 “是 否 归 还 ”等 于 1， 则 输出 “已 归还 ” 
S$ifbackstr=" 已 归还 "; 

2 

?> 

/* xxxxxxxxhtAhxhthxwhkhswhxhtasxtt 蜀 出 符合 查询 条 件 的 记录 sxsxxszxtxxxktwxxtstwxxsws il 

<tr> 


<td style="padding:5px:">&nbsp;<?php echo $result[barcode];?></td> 
<td style="padding:5px;"><a href="book_look.php?id=<?php echo $result[id]; ?>"><?php echo 
$result[bookname];?></a></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[readerbarcode];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $resultIname];?></td> 
<td style="padding:5px:">&nbsp;<?php echo $result[borrowTime];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $result[backTime];?></td> 
<td style="padding:5px;">&nbsp;<?php echo $ifbackstr;?></td> 
< 
he OR MEE RL 
<?php 
}while($result=mysql_fetch_array($sq)l)); 
} 


28.9 小 结 


本 章 运用 软件 工程 的 设计 思想 ， 通 过 一 个 完整 的 图 书馆 管理 系统 引导 读者 深入 了 解 系统 的 开发 流程 。 
在 该 系统 的 实现 过 程 中 ， 除 了 应 用 一 些 基本 的 PHP 技术 外 ， 还 涉及 一 些 独 特 的 技术 细节 ， 如 权限 设置 、 多 
表 查询 技术 等 ， 希 望 读 者 掌握 并 能 在 实际 的 操作 中 灵活 应 用 ， 举 一 反 三 。 


